【问题标题】:variadic templates with template function names for c++11带有 c++11 模板函数名称的可变参数模板
【发布时间】:2016-09-19 00:23:13
【问题描述】:

this 问题之后,我试图避免复制粘贴一些与调用BaseSensor 类的所有同名方法相关的代码。

在sensor.hpp中

struct EdgeSensor //a mixin
{
    void update(){}
    void printStats() {}
};

struct TrendSensor //another mixin
{
    void update(){}
    void printStats() {}
};

template<typename ... SensorType>
class BaseSensor : public SensorType ... //to my BaseSensor class
{
    void update() /*{ what goes in here??? }*/
    void printStats() /*{ what goes in here??? }*/
};

在sensor.t.hpp中

template<typename ... SensorType>
void BaseSensor<SensorType...>::update()
{
    int arr[] = { (SensorType::update(), 0)..., 0 };
    (void)arr;
}

template<typename ... SensorType>
void BaseSensor<SensorType...>::printStats()
{
    int arr[] = { (SensorType::printStats(), 0)..., 0 };
    (void)arr;
}

在 main.cpp 中

int main(int , const char **) 
{
    {
        BaseSensor<EdgeSensor,TrendSensor> ets;
        ets.update();
        ets.printStats();
    }
    {
        BaseSensor<EdgeSensor> ets;
        ets.update();
        ets.printStats();
    }
}

上面的代码依次执行所有mixin的update(),然后继续执行所有mixin的所有printStats()

我想知道是否有可能避免重复执行 BaseSensor::update()BaseSensor::printStats() 并创建一个通用(模板)函数,该函数接受目标函数的名称以在所有 mixins 中执行:

例如,我可以创建一个方法runAll()

template<typename ... SensorType>
class BaseSensor : public SensorType ... //to my BaseSensor class
{
    void update() /*{ what goes in here??? }*/
    void printStats() /*{ what goes in here??? }*/

    template<typename FnName>
    void runAll(FnName f)
    {
        int arr[] = { (SensorType::f(), 0)..., 0 };
        (void)arr;
    }
};

我将如何从BaseSensor::update()BaseSensor::printStats() 调用它。我尝试过使用

void update() { runAll<update>(); }
void printStats() { runAll<printStats>(); }

但这不起作用(没想到会这样)。将函数名称作为函数参数传递的问题(我看到许多其他问题,例如here 是我不知道如何从BaseSensor::update() 指向各种::update() 函数。例如

void update() { runAll<update>( update() ); }

也不正确。

在这种情况下是否可以避免复制?这可以在单行中完成,以避免使用 c++11 进行大量复制(即不使用通用 lambda,就像 here 所做的那样)?如果我将工作的runAll() 移动到文件“sensor.t.hpp”中,模板参数会是什么样子?

谢谢。

【问题讨论】:

  • 仅链接的答案不好,同样适用于问题。添加相关部分以自给自足。
  • @Jarod42 好的修复

标签: c++ templates c++11 variadic-templates variadic-functions


【解决方案1】:

只要要调用的函数是两个,就可以使用专用结构,依靠重载来解决。
它遵循一个最小的工作示例:

#include<iostream>

struct Executor {
    template<typename T>
    static void execute(int, T &t) {
        t.update();
    }

    template<typename T>
    static void execute(char, T &t) {
        t.printStats();
    }
};

struct EdgeSensor
{
    void update() { std::cout << "EdgeSensor::update" << std::endl; }
    void printStats() { std::cout << "EdgeSensor::printStats" << std::endl; }
};

struct TrendSensor
{
    void update() { std::cout << "TrendSensor::update" << std::endl; }
    void printStats() { std::cout << "TrendSensor::printStats" << std::endl; }
};

template<typename ... SensorType>
class BaseSensor : public SensorType ...
{
    template<typename T>
    void execute() {
        int arr[] = { (Executor::execute(T{}, static_cast<SensorType&>(*this)), 0)..., 0 };
        (void)arr;
    }

public:
    void update() {
        execute<int>();
    }

    void printStats() {
        execute<char>();
    }
};

int main() {
    BaseSensor<EdgeSensor,TrendSensor> ets;
    ets.update();
    ets.printStats();
}

如果您有两个以上的函数要调用,我想choice 技巧在这里很适用。

【讨论】:

  • hm 仍然有一些部分(在结构中)需要为每个其他类似的功能复制。看来唯一真正通用的方法是通用 lambdas...
  • 我会使用一些枚举/标签来调度而不是无意义的char/int
  • @Jarod42 我也会这样做。在我看来,int/char 技巧更简洁,仅此而已。
【解决方案2】:

您仍然可以手动编写通用 lambda 的(简化版本):

void update() {
    execute([](auto &t) { t.update(); });
}

变成这样

void update() {
    struct {
        template <typename T>
        void operator () (T& t) const { t.update(); }
    } updater;
    execute(updater);
}

【讨论】:

  • 您好,谢谢您提供的信息。对于复制粘贴来说,这似乎仍然“很长”。理想情况下,我可以以某种方式将结构“输出”并为t.update() 函数调用提供一个“变量”名称?谢谢
  • 当你确实要引入 second 函数时,这是相当棘手的。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2014-09-08
  • 2012-08-14
  • 2016-11-18
相关资源
最近更新 更多