在 C 中有一种方法可以使用 va_list 获取动态长度参数列表,如下所述: http://www.cprogramming.com/tutorial/c/lesson17.html
这很简单,但大多数情况下在 C++ 中不需要。我目前正在构建一个顶级包装器类来封装一些 Zend 功能。
无论如何,我确实需要从普通函数动态地调用如 printf 这样的动态函数。
我的意思是上面描述的例子的相反方式,这是我得到的如此 war ,也许这更好地解释了我的想法:
void ZendParams::setRequired( int &Var )
{
// save every var pointer to a stack
// and save the type with it. (if needed, does not seems to be)
// after call to ZendParams::fetch()
// the values are stored in the passed variables
this->_params.push_back( (void*) &Var );
this->_paramStr.append( 'i' );
}
size_t ZendParams::fetch()
{
if ( zend_parse_parameters(
ZEND_NUM_ARGS() TSRMLS_CC, ,
this->_paramStr.c_str(),
...
) !== FAILURE)
{
}
return 0;
}
所以我想动态调用 zend_parse_parameters。
基本思想是在列表中添加指向变量的指针,并将它们动态传递给 zend 函数(作为引用)。
我想也一定有办法用 va_list 做到这一点。
但是如何呢?
为了让它更简单,我使用这个例子:
list<int> myList;
myList.push_back(1);
myList.push_back(5);
myList.push_back(10);
myList.push_back(37);
myList.push_back(42);
double function avg( int num, ... )
va_list arguments;
int sum = 0;
va_start ( arguments, num );
for ( int x = 0; x < num; x++ )
{
sum += va_arg ( arguments, int );
}
va_end ( arguments );
return sum / num;
}
我想用列表中的所有数字调用 avg
。我从上面提到的教程中获取了示例函数,但它应该以非常简单的方式显示我的意思。
但是,我无法更改调用的函数,因为它是 zend 框架的一部分。
在 C 或 C++ 中有什么方法可以做到这一点吗?
我的第二种方法:
template <typename... Arguments>
size_t Zend::getMethodParams( string paramStr, Arguments... Args )
{
if ( zend_parse_parameters(
ZEND_NUM_ARGS() TSRMLS_CC, ,
paramStr.c_str(),
Args...
) == FAILURE)
{
return 0;
}
}
这样调用(获取定义的 3 个参数):
string My1stParam;
int My2ndParam;
int My3rdParam;
Zend::getMethodParams<string, int, int>( "sii", &My1stParam, &My2ndParam, &My3rdParam );
我认为这应该可行,但有一个难题: zend_parse_parameters 函数返回一个字符串的 2 个值(c 风格的字符串!): - 一个字符指针和 - 字符串长度。
所以我要么必须这样调用它:
char* My1stParam_str;
int My1stParam_length
int My2ndParam;
int My3rdParam;
Zend::getMethodParams<string, int, int>( "sii", &My1stParam_str, &My1stParam_length, &My2ndParam, &My3rdParam );
string My1stParam;
My1stParam.assign(My1stParam_str, My1stParam_length);
无论如何,这就是我想要阻止的。
或者我将不得不修改传递给 zend_parse_parameters 函数的参数列表以在内部执行这些附加步骤。
我希望至少可以这样调用它:
string My1stParam;
int My2ndParam;
int My3rdParam;
Zend::getMethodParams<string, int, int>( "sii", &My1stParam, &My2ndParam, &My3rdParam );
所以说清楚:参数在编译时是已知的,但是在后面的源代码中,函数调用在所有出现的地方都会有很大的不同。
最佳答案
我在 zend 框架中找到了解决这个问题的方法。我之前确实搜索过这样的解决方案,但它似乎没有很好的记录(n.m. 已经提到,没有像 va_list 函数那样的 zend 内部方式)。
但确实有一个!
对于其他遇到这个问题的人:
long arg;
zval ***args;
int i, argc = ZEND_NUM_ARGS( );
if (zend_parse_parameters(1 TSRMLS_CC, "l", &arg) == FAILURE) return;
array_init(return_value);
add_index_long(return_value, 0, arg);
if(argc>1) {
args = (zval ***)emalloc(argc * sizeof(zval **));
if(zend_get_parameters_array_ex(argc, args) == FAILURE) {
efree(args);
return;
}
for(i = 1; i < argc; i++) {
zval_add_ref(args[i]);
add_index_zval(return_value,i, *args[i]);
}
efree(args);
}
这就是解决方案;)
此代码段,发现于 http://docstore.mik.ua/orelly/webprog/php/ch14_07.htm将所有参数动态解析为 PHP 数组的 c 表示形式。
关于c++ - 运行时的动态函数调用(va_list),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15988277/