Я хочу иметь функцию, которая принимает указатель на функцию и пересылает все параметры, как указано самим типом указателя на функцию, например:
template < typename RET, typename ... ARGS >
auto Do1( RET(*ptr)(ARGS...), ARGS... args )
{
(*ptr)(std::forward<ARGS>( args )...);
}
int main ()
{
int i=4;
Do1( &Ex1, i );
Do1( &Ex2, i ); //fails!
Do1( &Ex3, i+1 ); // fails
}
Функции для вызова для обоих примеров:
void Ex1( int i){ std::cout << __PRETTY_FUNCTION__ << " " << i << std::endl; i=10;}
void Ex2( int& i){ std::cout << __PRETTY_FUNCTION__ << " " << i << std::endl; i=20;}
void Ex3( int&& i){ std::cout << __PRETTY_FUNCTION__ << " " << i << std::endl; i=30;}
Сбой в случае Ex2
и Ex3
просто при попытке два раза определить тип списка ARGS, и результат будет другим. Компилятор жалуется на:
main.cpp:57:22: error: no matching function for call to 'Do1(void (*)(int&), int&)'
Do1( &Ex2, i ); //fails!
^
main.cpp:33:10: note: candidate: 'template<class RET, class ... ARGS> auto Do1(RET (*)(ARGS ...), ARGS ...)'
auto Do1( RET(*ptr)(ARGS...), ARGS... args )
^~~
main.cpp:33:10: note: template argument deduction/substitution failed:
main.cpp:57:22: note: inconsistent parameter pack deduction with 'int&' and 'int'
После этого я попытался обойти проблему следующим подходом, так как я выбираю типы только один раз, выводя список ARGS, и снова пересылаю к промежуточной лямбде следующим образом:
template < typename RET, typename ... ARGS >
auto Do2( RET(*ptr)(ARGS...) )
{
return [ptr]( ARGS ... args )
{
std::cout << __PRETTY_FUNCTION__ << std::endl;
(*ptr)(std::forward<ARGS>(args)...);
};
}
int main ()
{
int i=4;
Do1( &Ex1, i );
Do1( &Ex2, i ); //fails!
Do1( &Ex3, i+1 ); // fails
Do2( &Ex1 )( i );
std::cout << "now i: " << i << std::endl;
std::cout << std::endl;
Do2( &Ex2 )( i );
std::cout << "now i: " << i << std::endl;
std::cout << std::endl;
Do2( &Ex3 )( i+1 );
std::cout << "now i: " << i << std::endl;
std::cout << std::endl;
}
В: Есть ли способ исправить первый подход в любом случае, чтобы избавиться от промежуточной лямбды здесь? А если нет, хорошо ли спроектирована промежуточная лямбда-решение, особенно со всеми «переадресационными» вещами, чтобы я не создавал копий или других неожиданных действий?
EDIT:
Это только сокращенный пример. Я не намерен писать копию std::invoke
. Так что в моем реальном мире кода гораздо больше можно сделать внутри самого метода Do
.
Важно получить необходимые типы из типа указателя функции, так как я должен выполнить некоторые проверки внутри Do
, которые относятся к типам, предоставляемым указателем функции, а не из данного дополнительного параметры, которые я предоставляю из кода пользователя для метода Do
.