场景
1. 一个动态库中有一个变参的函数,这个变参函数声明时声明了两参数,编译的时候使用了“最大化速度(/O2)”及“只适用于 __inline (/Ob1)”优化选项;
2. 在外部调用这函数时,获取到的变参值为空。也就是说出现异常。
例子:
//dll
std::string stringcon(const char* a, const char* b, ...){
std::string str = string("(") + a;
va_list args;
va_start( args, a );
do{
str = str + "+" + *(char**)args;
va_arg(args,const char);
}while(*(char**)args != 0);
str = str + ")";
va_end(args);
return str;
}
调用
std::string str2= stringcon("1111","2222 ","33333333",0);
获取到的str2值为“1111”
a. 把这函数实现移到exe的本地上,执行这移过来函数,结果正确。
b. 把dll的优化去年,设为“禁用(/Od)”,再编译,运行得到的结果正确。
c.接着把动态库接口改成如下,其它都不变:
std::string stringcon2(const char* a, ...){
std::string str = string("(") + a;
va_list args;
va_start( args, a );
do{
str = str + "+" + *(char**)args;
va_arg(args,const char);
}while(*(char**)args != 0);
str = str + ")";
va_end(args);
return str;
}
结果正确,结果为“1111+2222+33333333”。
d.接着做了第二次实验:
把函数改成va_start( args, b ),如:
std::string stringcon2(const char* a, const char* b, ...){
std::string str = string("(") + a;
va_list args;
va_start( args, b );
do{
str = str + "+" + *(char**)args;
va_arg(args,const char);
}while(*(char**)args != 0);
str = str + ")";
va_end(args);
return str;
}
结果为:1111+33333333
结果为预知的。
由上可知,vs在编译的时候对可变参数,如果有两个形参声明的,会优化。
解决方法
方法1.把优化去掉;
方法2.修改接口或者代码。