[解決済み】一致する関数ポインターを呼び出すためにタプルを「解凍」する
2022-04-03 05:34:19
質問
に格納しようとしています。
std::tuple
この値は、後で、格納された型に一致する関数ポインタの呼び出しの引数として使用されます。
私が苦労している問題を示す簡略化した例を作りました。
#include <iostream>
#include <tuple>
void f(int a, double b, void* c) {
std::cout << a << ":" << b << ":" << c << std::endl;
}
template <typename ...Args>
struct save_it_for_later {
std::tuple<Args...> params;
void (*func)(Args...);
void delayed_dispatch() {
// How can I "unpack" params to call func?
func(std::get<0>(params), std::get<1>(params), std::get<2>(params));
// But I *really* don't want to write 20 versions of dispatch so I'd rather
// write something like:
func(params...); // Not legal
}
};
int main() {
int a=666;
double b = -1.234;
void *c = NULL;
save_it_for_later<int,double,void*> saved = {
std::tuple<int,double,void*>(a,b,c), f};
saved.delayed_dispatch();
}
を含む問題では、通常
std::tuple
やバリアドテンプレートのような別のテンプレートを書きます。
template <typename Head, typename ...Tail>
を使用して、すべての型をひとつずつ再帰的に評価します。しかし、関数呼び出しをディスパッチするためにそれを行う方法は見当たりません。
本当の動機はもう少し複雑で、どうせほとんど学習用のエクササイズに過ぎない。私は他のインターフェースから契約によってタプルを渡されたので、変更することはできませんが、それを関数呼び出しにアンパックしたいという欲求は私のものであると仮定することができます。このため
std::bind
を、根本的な問題を回避するための安価な方法として利用することができます。
を使用して呼び出しをディスパッチするきれいな方法は何ですか?
std::tuple
あるいは、将来の任意の時点まで、いくつかの値と関数ポインタを保存/転送するという、同じ結果を得るための別のより良い方法はありますか?
どのように解決するのですか?
C++17の解決策は、単純に
std::apply
:
auto f = [](int a, double b, std::string c) { std::cout<<a<<" "<<b<<" "<<c<< std::endl; };
auto params = std::make_tuple(1,2.0,"Hello");
std::apply(f, params);
ただ、このスレッドの回答で一度記載されるべきだと思いました(すでにコメントの一つに出てきた後)。
このスレッドでは、C++14の基本的な解答がまだ見つかっていません。EDIT: いや、実はWalterの答えの中にあるのです。
この関数が与えられています。
void f(int a, double b, void* c)
{
std::cout << a << ":" << b << ":" << c << std::endl;
}
次のスニペットで呼び出します。
template<typename Function, typename Tuple, size_t ... I>
auto call(Function f, Tuple t, std::index_sequence<I ...>)
{
return f(std::get<I>(t) ...);
}
template<typename Function, typename Tuple>
auto call(Function f, Tuple t)
{
static constexpr auto size = std::tuple_size<Tuple>::value;
return call(f, t, std::make_index_sequence<size>{});
}
例
int main()
{
std::tuple<int, double, int*> t;
//or std::array<int, 3> t;
//or std::pair<int, double> t;
call(f, t);
}
関連
-
[解決済み】 unsigned int vs. size_t
-
[解決済み】getline()が何らかの入力の後に使用されると動作しない 【重複あり
-
[解決済み】C++のGetlineの問題(オーバーロードされた関数 "getline "のインスタンスがない
-
[解決済み】抽象クラス型の無効なnew-expression
-
[解決済み】Enterキーを押して続行する
-
[解決済み】'std::cout'への未定義の参照
-
[解決済み] スマートポインターとは何ですか?
-
[解決済み] なぜ、オブジェクトそのものではなく、ポインタを使用しなければならないのですか?
-
[解決済み] 関数呼び出しにおけるstarとdoublestarの演算子の意味は?
-
[解決済み】C言語の関数ポインタはどのように機能するのですか?
最新
-
nginxです。[emerg] 0.0.0.0:80 への bind() に失敗しました (98: アドレスは既に使用中です)
-
htmlページでギリシャ文字を使うには
-
ピュアhtml+cssでの要素読み込み効果
-
純粋なhtml + cssで五輪を実現するサンプルコード
-
ナビゲーションバー・ドロップダウンメニューのHTML+CSSサンプルコード
-
タイピング効果を実現するピュアhtml+css
-
htmlの選択ボックスのプレースホルダー作成に関する質問
-
html css3 伸縮しない 画像表示効果
-
トップナビゲーションバーメニュー作成用HTML+CSS
-
html+css 実装 サイバーパンク風ボタン
おすすめ
-
[解決済み】C++でint型に無限大を設定する
-
[解決済み】LLVMで暗黙のうちに削除されたコピーコンストラクタの呼び出し
-
[解決済み] error: 'ostream' does not name a type.
-
[解決済み】C++のGetlineの問題(オーバーロードされた関数 "getline "のインスタンスがない
-
[解決済み】C++プログラムでのコンソールの一時停止
-
[解決済み】リンカーエラーです。"リンカ入力ファイルはリンクが行われていないため未使用"、そのファイル内の関数への未定義参照
-
[解決済み】クラスのコンストラクタへの未定義参照、.cppファイルの修正も含む
-
[解決済み】 while(cin) と while(cin >> num) の違いは何ですか?)
-
[解決済み] template "と "typename "キーワードはどこに、なぜ入れなければならないのですか?
-
[解決済み] C++でコンパイル時の文字列を簡便に宣言する方法