运行时的动态函数调用(va_list)
Dynamic function calls at runtime (va_list)
在C语言中有一种使用va_list获取动态长度参数列表的方法,如下所述:http://www.cprogramming.com/tutorial/c/lesson17.html
非常简单,但在c++中大多数时候不需要。我目前正在构建一个顶级包装器类来封装一些Zend功能。
无论如何,我确实需要从一个普通函数动态地调用像 printf这样的动态函数。我的意思是与上面描述的例子相反的方式,这是我得到的结果,也许这能更好地解释我的想法:
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函数为字符串(c风格的字符串!)返回2个值:-一个char指针和-字符串长度
所以我要么这样称呼它:
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接受函数那样的内部方式)。
但确实有一个!
对于其他被这个问题困扰的人:
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表示。
- 使用std::multimap迭代器创建std::list
- 来自 std::list 的迭代器 .end() 按预期返回"0xcdcdcdcdcdcdcdcd"但 .begin()
- 如何在 C 中正确使用 libiconv 使其不会报告"Arg list too long"?
- C++中带有List类的迭代器Segfault
- 使用"std::unordereded_map"映射到"std::list"对象
- GCC对可能有效的代码抛出init list生存期警告
- 使用std::list创建循环链表
- 重载Singly Linked List中的赋值运算符
- '[](std::list& list)<int>{return std::move(list)}(list)' 是否保证将 'list' 留空?
- 如果 KEY 是 std::list 或 std::vector 而不是值,那么 std::map 的默认行为是什么?
- 为什么这个 std::queue/指向结构的指针列表直到 List.Size() == 0 才释放内存?
- "std::list::splice(std::const_iterator pos, std::list&& other)"是否保证将"其他"留空?
- 从嵌套循环中的 std::list 中删除将返回访问冲突
- QStringList vs list<shared_ptr<QString>> 性能比较C++
- 包含 std::list 的结构体的 C++ 初始化
- 在C++中使用 Catch 测试框架编译错误"error: expected ';' at end of declaration list"
- C++ assigment std::list:<typename>:itrator 在 main 中工作,但在方法中它不起作用
- 在基于范围的 for 循环期间插入 std::list 的后面
- 循环挂起迭代的 std::擦除 on std::list
- 如何增加以前由新运算符分配的 C++ std::list 数组的大小?