- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
阅读一些 C++ 代码时,我遇到了我称之为函数宏的“功能性”使用,大致如下(这是一个完全程式化的例子来说明这一点):
#define TOP_LEVEL(ARG1) \
ARG1("foo1","bar1") \
ARG1("foo2","bar2")
#define NEXT_LEVEL(ARG2A, ARG2B) \
cout << ARG2A << " and " << ARG2B;
TOP_LEVEL(NEXT_LEVEL)
我对这门语言比较陌生,一开始我想不通,但后来我只通过预处理器 (g++ -E
) 运行了它,你看它解析为:
cout << "foo1" << " and " << "bar1"; cout << "foo2" << " and " << "bar2";
你看到它在那里做了什么吗?它将宏 NEXT_LEVEL 像函数指针一样 传递给宏 TOP_LEVEL。看到这可能有多大用处,我想了解更多:将函数传递给其他函数是 pretty sophisticated stuff并且必须至少有更多关于该技术的说法。
然而,尽管进行了大量的谷歌搜索,我还是找不到证据表明预处理器的这个特性存在,更不用说任何接近文档的东西了:here , here , here和 here只是跳过这一步的宏教程的四个例子;最后一个甚至有一个名为“高级宏技巧”的部分 - 这肯定符合条件!?
(请注意,这与简单地将另一个求值函数宏作为参数调用函数宏完全不同——FOO(BAR(2)) 更为直接。)
我的问题是:
最佳答案
这个想法被创造为“X-Macro”。一些定义不会包含您的特定示例(X-macros 通常涉及更多,包含一个文件),但任何相关信息。 about this 将在搜索时属于该术语。
作为chris评论中提到,Boost.Preprocessor 使用了这个想法,效果很好。流行的用途是:BOOST_PP_REPEAT
、BOOST_PP_LIST_FOR_EACH
,最强大的是:BOOST_PP_ITERATE
。
BOOST_PP_ITERATE
是一个“真正的”X-Macro;包含单个文件会扩展为依赖于之前定义的宏的内容。我在这个 other answer 中展示了一个更“合适”的骨架框架, 但一个例子是:
// in xyz_data.def
DEFINE_XYZ(foo, 1, "Description A")
DEFINE_XYZ(bar, 5, "Description B")
DEFINE_XYZ(baz, 7, "Description C")
然后当我只想要第 1 列时,我可以这样做:
#define DEFINE_XYZ(name, number, desc) some_func(name)
#include "xyz_data.def"
在其他地方我想为每个函数生成一些函数,我可以这样做:
#define DEFINE_XYZ(name, number, desc) \
int BOOST_PP_CAT(get_number_for_, name)() \
{ \
std::clog << "Getting number, which is: " desc << std::endl; \
\
return number; \
}
#include "xyz_data.def"
然后您可以生成一个枚举,其中名称等于数字等。
强大的是,当我想添加一个新的 xyz 时,我只需将它添加到一个位置,它就会神奇地出现在任何需要它的地方。我在一个非常大的代码库中做过类似的事情,以便将一些书签数据保存在一个中央位置,但是各种属性在不同位置的使用方式不同。
请注意,通常没有办法解决这个问题;我所拥有的是在句法上不同,所以没有其他语言特性可以为我将它泛化到那个级别,只有宏。宏不是邪恶的。
您拥有的实际上是一个 X 宏,其中 .def 文件是独立的,足以成为 #define
。换句话说,#include "xyz_data.def"
只是 TOP_LEVEL
。
这只有一个很大的缺点,具有讽刺意味的是,这不是 X-macros 本身的使用,而是它们对 C 和 C++ 编译器的影响。问题在于预处理器允许我们在每次包含文件时更改文件的预处理结果,即使文件内容完全相同也是如此。
您可能听说过,与现代语言相比,C 和 C++ 的编译速度很慢,这就是原因之一。它没有合适的模块/打包系统,只是临时包含其他文件。我们刚刚了解到,一般来说这是无法避免的。哎呀。 (也就是说,编译器很聪明,例如,当您在一个文件周围包含保护措施时,它会注意到,并避免多次处理它。但这是视情况而定。)
也就是说,使用 X-Macros 本身不应该对实际程序的编译时间产生巨大影响。只是它们仅仅是潜在的存在就触及了真实的世界,并搅乱了编译器的头脑。
关于c++ - C++ 宏中的函数式编程风格 : Is this documented anywhere?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19672644/
Textmate 语法(.tmLanguage 文件)有时以 XML 格式表示。 我想转换为更易读的格式(即 JSON 或 YAML)以集成到 VS Code 语法突出显示扩展中。 为了澄清我的意思,
如何通过 pandas 样式隐藏列标签?有一个 hide_index() 方法可以删除索引行,不幸的是 hide_column() 标签会删除整个列(标题和数据)。我只想隐藏标题。谢谢! 最佳答案 s
我正在考虑为一组服务使用 SOA 架构来支持我咨询的业务,以前我们使用数据库集成,其中每个应用程序从共享的 MS SQL 数据库中挑选出它需要的东西并使用它等等。我们有各种与怪物数据库(包括 java
所以我有以下代码,我想知道 Objective-C 中哪种“风格”被认为更好。 选项 1: id temp = [dictionary objectForKey: @"aBooleanValue"];
当创建一个没有类参数的对象时,我很难决定是否应该包含空括号。一个具体的例子:我正在与现有的 Java 代码交互,并创建一个实现名为 EventResponder 的接口(interface)的对象。我
我有一个抽象类Stack和一个扩展它的类:MyStack。我需要为 MyStack 创建一个复制构造函数。只传入 MyStack 对象更好,还是传入任何 Stack 对象更好? public MySt
我正在考虑将那些在函数体中未修改的 Python 函数参数拼写为 ALL_UPPERCASE,向此类 API 的用户发出信号,表明传递的值不会被修改(如果一切都如广告所言,无论如何) )。我不知道这会
我的 build.gradle 文件、staging、stable 和 production 以及默认构建类型 debug 和 release。对于其中的每一个,我都有不同的 AAR 文件,例如,我有
假设我有以下文件: main.cpp 例程.cpp 例程.h 进一步假设 main.cpp 调用了在 routine.cpp 中定义的函数 routine(),但是 routine.cpp 还包含仅由
我对此进行了一些搜索,但实际上我还没有找到 MySQL 中用于创建外键的样式概念是什么 - 在创建表定义中或在 alter 语句中。谢谢。 最佳答案 何时创建外键: 如果在创建表时明确需要外键,则在创
您好,我正在尝试将 Android 应用风格(免费且完整)实现为动态壁纸。在 Eclipse 中,我曾经使用以下代码从我自己的 Android Activity 打开动态壁纸预览: I
我的 Android 应用程序有两种不同的风格,lite 和 pro。在应用程序中,我有一个名为 customFragment.java 的类,它包含在 main 中(不同风格之间没有区别)并且还包含
我有一个包含多个子目录的项目,如下所示: /opt/exampleProject/src ├── __init__.py ├── dir1 │ ├── __init__.py │ ├──
假设我们有类似的东西 int f(int n); .... do{ int a = b; int b = f(a); } 这样说有没有风险 do{ int b = f(b);
是否有风格指导或理由来选择其中一种模式而不是另一种? 最小化上下文管理器下的代码量“感觉”更干净,但我无法指出具体原因。这可能只是偏好,并没有关于此事的官方指导。 1) 里面的所有代码都有上下文。 w
module Hints module Designer def self.message "Hello, World!" end
我正在开发一个具有多种风格的 android 项目。 这很好用,我可以自定义应用程序的元素,例如颜色和字符串资源。 我想让一些风格基于 AppCompat 浅色主题,一些基于 AppCompat 深色
因此,这不起作用,因为 seatsAvailable 是最终的。如何使用更多的 lambda 风格的从头开始的方式来完成我想要完成的事情? final boolean seatsAvailable =
考虑以下代码: cpu_set_t cpuset; CPU_ZERO(&cpuset); CPU_SET(0, &cpuset); sched_setaffinity(0, sizeof(cpuset
从历史上看,我总是这样编写我的异常处理代码: Cursor cursor = null; try { cursor = db.openCursor(null, null
我是一名优秀的程序员,十分优秀!