- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有两个重载函数,唯一的区别是最后一个参数,在一种情况下是 va_list
在另一种情况下是省略号。
我注意到选定的重载是不同的,这取决于最后一个参数是整数文字 0 还是 1。
我制作了一个简化的工作示例,如下所示,我正在尝试各种输入并发现其他一些奇怪的场景:
#include <iostream>
#include <cstdarg>
void Func(va_list /*arg*/)
{
std::cout << "void Fun(va_list);" << std::endl;
}
void Func(...)
{
std::cout << "void Fun(...);" << std::endl;
}
int main()
{
std::cout << "Func(0);" << " >> ";
Func(0);
std::cout << "Func(1);" << " >> ";
Func(1);
std::cout << "Func(-1);" << " >> ";
Func(-1);
std::cout << "Func('0');" << " >> ";
Func('0');
std::cout << "Func('\\0');" << " >> ";
Func('\0');
std::cout << "Func(\"\");" << " >> ";
Func("");
std::cout << "Func(\"Dummy\");" << " >> ";
Func("Dummy");
std::cout << "Func(nullptr);" << " >> ";
Func(nullptr);
std::cout << "Func(0*1);" << " >> ";
Func(0*1);
std::cout << "Func(0.0);" << " >> ";
Func(0.0);
std::cout <<"int i = 0; Func(i);" << " >> ";
int i = 0;
Func(i);
std::cout <<"i = 1; Func(i);" << " >> ";
i = 1;
Func(i);
static const int j = 0, k = 1;
std::cout << "static const int j = 0; Func(j);" << " >> ";
Func(j);
std::cout << "static const int k = 1; Func(k);" << " >> ";
Func(k);
}
在
cppshell 中编译此代码使用 C++14 给出以下输出:
Func(0); >> void Func(va_list);
Func(1); >> void Func(...);
Func(-1); >> void Func(...);
Func('0'); >> void Func(...);
Func('\0'); >> void Func(va_list);
Func(""); >> void Func(...);
Func("Dummy"); >> void Func(...);
Func(nullptr); >> void Func(va_list);
Func(0*1); >> void Func(va_list);
Func(0.0); >> void Func(...);
int i = 0; Func(i); >> void Func(...);
i = 1; Func(i); >> void Func(...);
static const int j = 0; Func(j); >> void Func(...);
static const int k = 1; Func(k); >> void Func(...);
如您所见,一些调用解析为
va_list
重载,而大多数解析为省略号重载。
va_list
通常是
char*
.在 Visual C++ 中,我发现它定义在
vadefs.h
如
typedef char* va_list;
.但是,我仍然不明白这里的大多数情况下重载决议是如何工作的。
va_list
之间工作和省略号? Func(0)
解决省略号重载? 最佳答案
...
只有在没有隐式转换序列到 va_list
时才会选择重载.换句话说,省略号在重载解析期间始终具有最低的优先级。
自定义va_list
未指定,通常您观察到的结果将不可移植:Func(x)
可以拨打 va_list
重载或省略号重载取决于实现。唯一的异常(exception)是 Func(x)
将始终调用 va_list
过载时 x
本身有类型 va_list
(可能是 const 限定的)。
如果,事实上,va_list
是 char*
在您的系统上,可能会发生空指针转换:换句话说,0 可以隐式转换为类型 char*
的空指针。 .没有其他整数值可以隐式转换为指针值。
从 C++14 开始,甚至 '\0'
不应隐式转换为指针值。空指针转换仅适用于值为零的整数文字。 (尽管 '\0'
具有整数类型,但它不是整数文字。)此特定版本的 Visual Studio 已过时。
至于您的问题“我如何调用 Func(0) 来解析省略号重载?”,好吧,您无法更改重载解析的规则。您可以做的是将省略号重载更改为模板:
template <class T>
void Func(T x);
这保证了
va_list
仅当参数为
va_list
时才会调用重载(可能是 const 限定的)并且在所有其他情况下,模板将被调用,因为它将给出完全匹配。然而,一个问题仍然存在:你可能有一些
char*
变量,您认为它不是
va_list
,并调用
Func
以它为参数可以调用
va_list
过载自
va_list
是
char*
(在您的系统上)。没有办法阻止这种情况。也许你不应该重载
Func
根本。您可以改为使用两个单独的功能。
关于c++ - va_list 和省略号之间的重载分辨率因字面值而异。为什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/67793445/
尝试编译此代码时 #include void bar_ptr(int n, va_list *pvl) { // do va_arg stuff here } void bar(int n,
假设我有一个函数,它接受可变参数 (...) 或从另一个此类函数传递的 va_list 。主要逻辑在这个函数本身(我们称之为 f1),但我想让它将 va_list 传递给另一个函数(我们称之为 f2)
看似简单的问题,但一直未能在 SO 或 C 标准中找到答案。问题是 va_list 是否必须是 C 中函数的最后一个参数(当然,假设该函数将 va_list 作为参数)。 例如,这样安全吗? int
我正在尝试打印字符串类型的参数变量,但我一直卡在这上面 想法 尝试使用 _Str[i] 作为 while 循环内的表达式在 srting 内部移动,一旦 _Str[i] == '\0' 返回 (-1
下面的 max 函数应该返回 5,但它返回的是 4294967294。我怀疑奇怪的行为是由变量转换引起的,但无法弄清楚。有人能检测出故障吗? 系统:Windows 7(64位),mingw64 #in
在 Fedora 11 上编译一些工作代码时,我收到此错误: /usr/include/c++/4.4.1/cstdarg:56: error: ‘::va_list’ has not been de
全部, 我想控制 va_list 中的传递参数的数量。 va_list args; va_start(args, fmts); vfprintf(stdout, fmts, args)
我有一个很好的 C++ 函数,它通过 va_list + va_start + va_arg 支持多个参数。但我必须将我的项目转换为 C。转换后,这个 va_list 构造被编译器拒绝(多个错误)。有
我已经编写了这段代码,但我不确定它是否可以。 size_t sendHeaders(int fd, int seconds, const char* header1, ...) { size_
我无法找到我的“问题”的答案。 我创建了一个函数,它接受不同数量的整数('findMinVal(int x, ...)')并返回所有调用中的最小数字。现在我的程序直接通过代码获取参数: int mai
我一直在追踪一个问题,并将范围缩小到这个框架: #include #include #include #include #include #include typedef struct _
我有一个主要更新,我在通话前进行访问控制。我想在我的代码中使用一个地方来调用任何紧急函数。 我有一个结构 ACTION: { FUNC_PROTOTYPE pfnAction; uns
我目前有 2 个函数重载: void log(const char* format, ...); void log(const string& message); 我希望在这个调用的情况下:log("
在 case 's' 下,我得到一个坏指针 (0xcccccccc) 异常: string Logger::format(const char *str, va_list args) { os
#include #include void s(const char* param, ...) { va_list arguments; va_start (arguments,
我试着写一个这样的函数: int solve(double* x, double xA, double xB, double zeps, double funct(double x
我有这样的代码 #include "atlstr.h" void DisplayMessage(CString pszFormat, ...) { CString str; va_l
我正尝试在 C++ 中为一个项目实现一种反射形式。这个想法是,您将带有标签的类作为一种模板注册到映射中,然后调用共享基类的 cloneNew 方法来实际创建您想要的对象。但是,当我尝试使用 va_li
我已经(简单地)编写了这段代码。但它需要两次修复才能确定。 size_t send_header(int fd, int seconds, const char* header1, ...) {
我有一个用于微 Controller 的以太网库(用于 lpc2478 的 keil rl-tcpnet)。库以这种方式在调试输出函数中使用 va_list(指向 stdarg.h 中定义的 arg
我是一名优秀的程序员,十分优秀!