Cross platform C++ code architecture

前端 未结 8 616
广开言路
广开言路 2021-01-31 00:18

I\'m having a bit of a go at developing a platform abstraction library for an application I\'m writing, and struggling to come up with a neat way of separating my platform indep

8条回答
  •  醉话见心
    2021-01-31 00:59

    I might go for a policy-type thing:

    template
    struct PlatDetails : private Platform {
        std::string getDetails() const {
            return std::string("MyAbstraction v1.0; ") + getName();
        }
    };
    
    // For any serious compatibility functions, these would
    // of course have to be in different headers, and the implementations
    // would call some platform-specific functions to get precise
    // version numbers. Using PImpl would be a smart idea for these 
    // classes if they need any platform-specific members, since as 
    // Joe Gauterin says, you want to avoid your application code indirectly
    // including POSIX or Windows system headers, containing useless definitions.
    struct Windows {
        std::string getName() const { return "Windows"; }
    };
    
    struct Linux {
        std::string getName() const { return "Linux"; }
    };
    
    #ifdef WIN32
        typedef PlatDetails PlatformDetails;
    #else
        typedef PlatDetails PlatformDetails;
    #endif
    
    int main() {
        std::cout << PlatformDetails().getName() << "\n";
    }
    

    There's not a whole lot to choose though between doing this, and doing regular simulated dynamic binding with CRTP, so that the generic thing is the base and the specific thing the derived class:

    template
    struct PlatDetails {
        std::string getDetails() const {
            return std::string("MyAbstraction v1.0; ") + 
                static_cast(this)->getName();
        }
    };
    
    struct Windows : PlatDetails {
        std::string getName() const { return "Windows"; }
    };
    
    struct Linux : PlatDetails {
        std::string getName() const { return "Linux"; }
    };
    
    #ifdef WIN32
        typedef Windows PlatformDetails;
    #else
        typedef Linux PlatformDetails;
    #endif
    
    int main() {
        std::cout << PlatformDetails().getName() << "\n";
    }
    

    Basically in the latter version, getName must be public (although I think you can use friend) and so must be the inheritance, whereas in the former, the inheritance can be private and/or the interface functions can be protected, if desired. So the adaptor can be a firewall between the interface the platform has to implement, and the interface your application code uses. Furthermore you can have multiple policies in the former (i.e. multiple platform-dependent facets used by the same platform-independent class), but not for the latter.

    The advantage of either of them over versions with delegates or non-template-using inheritance, is that you don't need any virtual functions. Arguably this isn't a whole lot of advantage, considering how scary both policy-based design and CRTP are at first contact.

    In practice, though, I agree with quamrana that normally you can just have different implementations of the same thing on different platforms:

    // Or just set the include path with -I or whatever
    #ifdef WIN32
        #include "windows/platform.h"
    #else
        #include "linux/platform.h"
    #endif
    
    struct PlatformDetails {
        std::string getDetails() const {
            return std::string("MyAbstraction v1.0; ") + 
                porting::getName();
        }
    };
    
    // windows/platform.h
    namespace porting {
        std::string getName() { return "Windows"; }
    }
    
    // linux/platform.h
    namespace porting {
        std::string getName() { return "Linux"; }
    }
    

提交回复
热议问题