Динамические вызовы функций во время выполнения (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 Framework.

Есть ли способ в 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 ответ

Решение

Я нашел способ обойти это в рамках Zend. Я действительно уже искал такое решение раньше, но, похоже, оно не очень хорошо задокументировано (nm уже учил, что нет внутреннего способа 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 динамически разбирает все параметры в представлении c массива PHP.

Другие вопросы по тегам