草庐IT

C++11 - 编译时多态性解决方案

coder 2024-02-18 原文

假设我正在编写一个跨平台库,我必须以不同平台有不同行为的方式组织代码,并且这种行为(或定义)是在编译时根据平台选择的我的图书馆正在编译。

在 C++ 中执行此操作的“通常”方法是在编写方法或类时用大量 #ifdef 污染代码。

方法的问题在于:

  • 源代码看起来真的很丑
  • 如果您支持 3 个平台,您的源代码大约比您真正需要的大 3 倍,这意味着您的编译器仍然需要解析和分析所有代码才能“看到”#ifdef
  • 不同的实现之间没有真正的区别,当代码库增长时很难维护,而当你只有 3-4 个平台时,它增长得非常快。

由于 C++11 中有很多新特性,我想知道是否有什么改变,是否有新的选项。

最佳答案

您应该使用您的构建系统来执行此操作。您应该为 header 提供独立于平台的函数声明和类定义。然后,根据目标平台,构建系统应该编译这些函数和类的适当实现。

例如,让我们考虑创建用于显示图形或 GUI 元素的窗口。如果您不使用库来执行此操作,则必须自己编写跨平台代码。首先,您应该确切地考虑平台无关的接口(interface)应该是什么。也许你有一个 window类和一些辅助函数。然后,您可以在头文件中提供该类的定义和辅助函数的声明,并为每个平台提供单独的实现。然后你会有一组这样的文件:

  • window.h
  • window_wayland.cpp
  • window_winapi.cpp
  • window_x11.cpp

现在,所有需要使用您的类和函数的文件都应该是 #include <window.h> .他们都得到相同的函数声明。但是,您在构建系统的配置中指定 window_x11.cpp应该在带有 X11 窗口系统的系统上编译,window_wayland.cpp在带有 Wayland 和 window_winapi 的系统上在 Windows 上。这意味着,根据您构建的平台,您将获得适用于目标平台的该 header 的实现。

这有几个优点:

  1. 您已将构建问题(您正在为哪个平台构建)与代码问题分开。
  2. 每个平台相关的实现都有自己的文件。
  3. 您的文件中没有乱七八糟的预处理器指令并且难以遵循执行路径。

这并不意味着使用定义有选择地编译代码的不同部分有任何问题。我更喜欢只使用少量已本地化到平台相关部分的代码来查看这一点。理想情况下,将依赖于平台的代码包装在一个函数中并具有 #ifdef我们只是换掉实现。

具体如何进行这种选择性构建取决于您使用的构建系统。对于GNU构建系统,可以通过automake实现条件编译。 Some examples are given in the documentation .一个简单的例子是:

bin_PROGRAMS = hello
if LINUX
hello_SOURCES = hello-linux.c hello-common.c
else
hello_SOURCES = hello-generic.c hello-common.c
endif

正在运行 automake使用此配置将生成适当的 makefile。

关于C++11 - 编译时多态性解决方案,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15849419/

有关C++11 - 编译时多态性解决方案的更多相关文章

  1. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  2. ruby - 在 jRuby 中使用 'fork' 生成进程的替代方案? - 2

    在MRIRuby中我可以这样做:deftransferinternal_server=self.init_serverpid=forkdointernal_server.runend#Maketheserverprocessrunindependently.Process.detach(pid)internal_client=self.init_client#Dootherstuffwithconnectingtointernal_server...internal_client.post('somedata')ensure#KillserverProcess.kill('KILL',

  3. ruby - Sinatra set cache_control to static files in public folder编译错误 - 2

    我不知道为什么,但是当我设置这个设置时它无法编译设置:static_cache_control,[:public,:max_age=>300]这是我得到的syntaxerror,unexpectedtASSOC,expecting']'(SyntaxError)set:static_cache_control,[:public,:max_age=>300]^我只想将“过期”header设置为css、javaascript和图像文件。谢谢。 最佳答案 我猜您使用的是Ruby1.8.7。Sinatra文档中显示的语法似乎是在Ruby1.

  4. ruby - 使用 `+=` 和 `send` 方法 - 2

    如何将send与+=一起使用?a=20;a.send"+=",10undefinedmethod`+='for20:Fixnuma=20;a+=10=>30 最佳答案 恐怕你不能。+=不是方法,而是语法糖。参见http://www.ruby-doc.org/docs/ProgrammingRuby/html/tut_expressions.html它说Incommonwithmanyotherlanguages,Rubyhasasyntacticshortcut:a=a+2maybewrittenasa+=2.你能做的最好的事情是:

  5. ruby - 安装libv8(3.11.8.13)出错,Bundler无法继续 - 2

    运行bundleinstall后出现此错误:Gem::Package::FormatError:nometadatafoundin/Users/jeanosorio/.rvm/gems/ruby-1.9.3-p286/cache/libv8-3.11.8.13-x86_64-darwin-12.gemAnerroroccurredwhileinstallinglibv8(3.11.8.13),andBundlercannotcontinue.Makesurethat`geminstalllibv8-v'3.11.8.13'`succeedsbeforebundling.我试试gemin

  6. 安卓apk修改(Android反编译apk) - 2

    最近因为项目需要,需要将Android手机系统自带的某个系统软件反编译并更改里面某个资源,并重新打包,签名生成新的自定义的apk,下面我来介绍一下我的实现过程。APK修改,分为以下几步:反编译解包,修改,重打包,修改签名等步骤。安卓apk修改准备工作1.系统配置好JavaJDK环境变量2.需要root权限的手机(针对系统自带apk,其他软件免root)3.Auto-Sign签名工具4.apktool工具安卓apk修改开始反编译本文拿Android系统里面的Settings.apk做demo,具体如何将apk获取出来在此就不过多介绍了,直接进入主题:按键win+R输入cmd,打开命令窗口,并将路

  7. ruby - 如何计算 Liquid 中的变量 +1 - 2

    我对如何计算通过{%assignvar=0%}赋值的变量加一完全感到困惑。这应该是最简单的任务。到目前为止,这是我尝试过的:{%assignamount=0%}{%forvariantinproduct.variants%}{%assignamount=amount+1%}{%endfor%}Amount:{{amount}}结果总是0。也许我忽略了一些明显的东西。也许有更好的方法。我想要存档的只是获取运行的迭代次数。 最佳答案 因为{{incrementamount}}将输出您的变量值并且不会影响{%assign%}定义的变量,我

  8. arrays - Ruby 数组 += vs 推送 - 2

    我有一个数组数组,想将元素附加到子数组。+=做我想做的,但我想了解为什么push不做。我期望的行为(并与+=一起工作):b=Array.new(3,[])b[0]+=["apple"]b[1]+=["orange"]b[2]+=["frog"]b=>[["苹果"],["橙子"],["Frog"]]通过推送,我将推送的元素附加到每个子数组(为什么?):a=Array.new(3,[])a[0].push("apple")a[1].push("orange")a[2].push("frog")a=>[[“苹果”、“橙子”、“Frog”]、[“苹果”、“橙子”、“Frog”]、[“苹果”、“

  9. ruby-on-rails - rails 多态关联(遗留数据库) - 2

    我使用的是遗留数据库,所以我无法控制数据模型。他们使用了很多多态链接/连接表,就像这样createtableperson(per_ident,name,...)createtableperson_links(per_ident,obj_name,obj_r_ident)createtablereport(rep_ident,name,...)其中obj_name是表名,obj_r_ident是标识符。因此链接的报告将按如下方式插入:insertintoperson(1,...)insertintoreport(1,...)insertintoreport(2,...)insertint

  10. Ruby 守护进程和 JRuby - 备选方案 - 2

    我有一个应用程序正在从Ruby迁移到JRuby(由于需要通过Java提供更好的Web服务安全支持)。我使用的gem之一是daemons创建后台作业。问题在于它使用fork+exec来创建后台进程,但这对JRuby来说是禁忌。那么-是否有用于创建后台作业的替代gem/wrapper?我目前的想法是只从shell脚本调用rake并让rake任务永远运行......提前致谢,克里斯。更新我们目前正在使用几个与Java线程相关的包装器,即https://github.com/jmettraux/rufus-scheduler和https://github.com/philostler/acts

随机推荐