- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
背景:我的代码(我无法在此处发布)最终将在微 Controller 上运行,并且宏仅提供一种通过 1 个宏定义机制创建多个引脚定义函数的方法。我使用 windows 和 gcc 来尝试这些。
我尝试尽可能地抽象问题。我使用 std 控制台函数是因为我可以方便地将其显示在控制台窗口中。因此,我还将文件另存为 .cpp 并在 Windows 上使用 g++ 进行编译。
假设我这样设置代码:
#define MACRO2(_x) foo##_x(_x)
#define MACRO1(_x) MACRO2(_x)
#define BAR 3
void fooBAR(int num)
{
std::cout << num << std::endl;
}
如果我运行以下代码(工作示例)
int main()
{
MACRO2(BAR);
return 0;
}
第一个 BAR 被插入到 ##_x 中,从而定义要调用的函数名称,然后 BAR 作为该函数的参数插入并扩展为其值,因此我们得到 fooBAR(3)。代码有效,没有错误。
现在,如果我尝试在两者之间添加一个宏(这是我由于无法深入的原因而面临的现实世界情况),我的代码如下所示:
int main()
{
MACRO1(BAR);
return 0;
}
但是此代码会引发错误,因为当 MACRO1(BAR) 被 MACRO2(BAR) 替换时,(BAR) 会扩展为 3,而 MACRO2(3) 会导致未定义的 foo3(3),错误日志确认:
error: 'foo3' was not declared in this scope
所以要求是:
- I need to pass BAR into MACRO1 and it needs to be passed to MACRO2 without being expanded
- The word BAR has to stay exactly as it is, I know I could use ## in order to prevent it from expanding, but then I would need to add a char to BAR and the function call wouldn't work anymore.
有可能以某种方式完成这件事吗?将一个宏作为参数传递给另一个宏,而不在此过程中扩展初始宏?
最佳答案
But this code throws an error, because when MACRO1(BAR) gets substituted with MACRO2(BAR), (BAR) then gets expanded into 3, and MACRO2(3) leads to foo3(3)
是的。这是为您的特定宏集指定的预处理器行为。
在识别出类函数宏的参数后,它们在被替换到宏的替换文本之前会被完全宏扩展,除非它们是 ##
或 的操作数#
预处理器运算符。评估这些运算符的任何外观,然后重新扫描生成的文本以及适当的任何后续文本,以扩展其他宏。
Is it possible to somehow get this done? Pass a macro to another macro as an argument, without the initial macro being expanded in the process?
仅当参数是 ##
或 #
运算符的操作数时。后者对您没有帮助,但前者提供了一种解决方法:您可以传递一个附加的空参数,以便您可以在不更改所需参数的情况下执行串联:
#define MACRO2(_x) foo##_x(_x)
#define MACRO1(_x,dummy) MACRO2(_x##dummy)
#define BAR 3
int main()
{
MACRO1(BAR,);
return 0;
}
扩展为
int main()
{
fooBAR(3);
return 0;
}
如果你想避免额外的逗号,那么你可以通过设置 MACRO1
变量来实现:
#define MACRO2(_x) foo##_x(_x)
#define MACRO1(_x,...) MACRO2(_x##__VA_ARGS__)
#define BAR 3
int main()
{
MACRO1(BAR);
return 0;
}
它扩展到与另一个相同的东西。
请注意,这两种方法都有可能通过向顶级宏提供不需要的额外参数值来引入错误。人们可能会认为大多数此类错误会在编译时被捕获,因为扩展会导致代码损坏,就像问题中的尝试一样。但很难排除这样的可能性:这种错误会巧合地扩大到碰巧有效但错误的事情。
关于c++ - 您可以将一个宏作为参数提供给另一个宏,而不扩展初始宏吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57870022/
我开发了一个具有基本安全性的 Spring Boot 应用程序。我有两个具有相同路径和不同 http 方法的端点。当我使用默认密码/使用 application.yml 中给出的密码包含基本安全性时,
我的代码是这样的: 或者,像这样: 如果我首先列出 webm 源,Firefox 4 会播放它,但 Firefox 3.6 也会尝试播放它(但会失败,因为它不支持 webm)。
我希望提供一个泛型类型作为类型参数而不首先将其解析为具体类型。换句话说,我正在寻找一种方法来指定从基类继承时可以使用的类型映射函数。 示例(不正确的)语法,希望比我能解释得更好: abstract c
我在 .NET 中编写了一些桌面应用程序,它们既提供了用于正常使用的前端 GUI,也提供了用于其他需求(例如扩展、调度、自动化、高级使用等)的命令行界面。命名两个可执行文件的最佳做法是什么,因为它们构
我最近在这里思考了很多关于屏幕抓取以及它可能是一项什么样的任务。所以我提出以下问题。 作为网站开发人员,您是否会公开简单的 API 以防止用户抓取屏幕,例如 JSON 结果? 然后这些结果可以实现缓存
我正在为一个项目使用 Dojo 1.9,但我不明白 dojo.provide 的正确替代方案与传统风格相比,AMD 风格。我正在阅读 this文档页面。 很明显,这就是旧语法映射到新语法的方式: 旧
我正在开发一个 Angular 应用程序。当我使用 ng serve 正常运行它时,它运行没有任何错误.但是,当我运行 ng build --prod ,它给出了以下错误。 ERROR in Ille
我有一个 Mac 应用程序。在我的 Mac 应用程序中,我的屏幕之一有一个包含文本字段的 scrollView。在同一屏幕上,我有一个需要提供打印选项的按钮。可以打印文本字段的文本。打印按钮应调用 M
我已经成功地为普通媒体文件提供媒体文件,但是当我尝试提供管理媒体文件时,我失败了。请帮我找出问题所在,因为我已经尝试解决问题几个小时但没有运气(也一直在谷歌搜索并阅读有关提供静态文件的 django
我正在尝试创建一个简单的错误处理项目,它会在收到错误(例如 404、422 或 500)后为 JSON 提供错误数据。我使用来自 this 的代码网站,但它不适合我。 我实际上有这两个类: 基本 Co
假设我有一个名为 Number 的类(class),我打算对 Number 进行大量相等比较对象。我担心通用 Number::equals(Object o) 的“开销”(类比较等...)方法。在这种
假定以下情况: 对等方A只希望将音频流发送给对等方B 对等B只希望将视频流发送给对等A 从而, 与创建报价 var sdpConstraints = { “必填”:{ 'OfferToReceiveA
因为我有一些角度,所以我想检查角度模数 360°: double angle = 0; double expectedAngle = 360; angle.Should().B
这是我的程序中构建的 monad 堆栈: type Px a = ReaderT PConf (State PState) a 其中 PConf 和 PState 是保存应用程序的配置和状态的任意数据
因为我有一些角度,所以我想检查角度模数 360°: double angle = 0; double expectedAngle = 360; angle.Should().B
我有一个小程序需要以某些权限运行,这意味着加载时会显示一条警告消息。如果用户拒绝警告消息,我想重定向到错误页面并解释发生了什么。有什么办法可以做到这一点吗? 我研究过让计时器运行并在特定时间段后重定向
从我可以从 Firebase 文档中推断出,似乎需要服务器来提供静态内容(html和 javascript),所以你需要有一台托管机器和一个静态内容服务器在某处启动并运行,或某些服务托管静态站点。 对
关闭。这个问题是opinion-based 。目前不接受答案。 想要改进这个问题吗?更新问题,以便 editing this post 可以用事实和引文来回答它。 . 已关闭 3 年前。 Improv
我的项目根目录的静态文件夹中有一个文本文件。 我想提供它,所以我创建了: @csrf_exempt def display_text(request): content = retur
我目前正在研究指针,为了进一步理解我正在尝试使用指针将两个数值数组连接成一个。代码如下所示。 #include void concat(int **pa,int **pb,int **pc) {
我是一名优秀的程序员,十分优秀!