运行时调用动态函数(va_list)

时间:2013-04-13 13:23:42

标签: c++ c variadic-functions dynamic-function

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。

基本思想是在列表中添加指向vars的指针,并将它们动态地传递给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 );

所以说得清楚:参数在编译时是已知的,但函数调用在后面的源代码中的所有实例中都会非常不同。

1 个答案:

答案 0 :(得分:0)

我在zend框架中找到了解决这个问题的方法。我之前确实在搜索过这样的解决方案,但它似乎没有很好的文档记录(现在已经有了精心设计,没有像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表示。