如何访问 C++ 模板参数包中的类型?

How can I access the types in a C++ template parameter pack?

我想构造一个 class,表示从类型 I 的值到类型 O 的值的映射。 我创建了这个界面:

template <typename I, typename O> class Transform {
public:
    virtual O transform(I i) = 0;
};

接下来,我想让 Transform 个对象组合成一种 "pipe lined" 转换:

template <typename X, typename Y, typename Z> class MergeTransform : public Transform<X, Z> {
private:
    Transform<X, Y> *first;
    Transform<Y, Z> *second;
public:
    MergeTransform(Transform<X, Y> *first, Transform<Y, Z> *second) {
        this->first = first;
        this->second = second;
    }
    Z transform(X x) {
        return second->transform(first->transform(x));
    }
};

目前困扰我的是试图弄清楚如何使用模板来合并任意数量的转换。我想做类似

的事情
template <typename A, typename B, typename C> Transform<A, C> *merge(Transform<A, B> *t1, Transform<B, C> *t2) {
    return new MergeTransform(t1, t2);
}
template <typename A, typename... Bs, typename C> Transform<A, C> *merge(/* what should go here? */) {
    // ...
}

但是我不知道如何在第二个 merge 方法中表示来自 A->B1, B1->B2, B2->...->Bn, Bn->CTransform 的列表。 我还考虑过让模板参数成为转换本身,即。

/*
 * assume A and B are Transforms
 */
template <typename A, typename B> /* some return Transform type here */ *merge(A *a, B *b) {
    return new MergeTransform(a, b);
}
template <typename A, typename... Bs, typename C> /* return type */ *merge(A *a, Bs... bs, C *c) {
    return merge(merge(a, bs), c);
}

但如果不知道 A 和 B Transform 的模板化类型,就无法定义 return 类型。有没有办法访问这些值?理想情况下,我可以说

template <Transform<X, Y> A, Transform<Y, Z> B> Transform<X, Z> *merge(A *a, B *b) {
    return new MergeTransform(a, b);
}
template <Transform<W, X> A, Transform<?, ?>... Bs, Transform<Y, Z> C> Transform<W, Z> *merge(A *a, Bs... *bs, C *c) {
    return merge(merge(a, bs), c);
}

如果 merge(a, bs) 的扩展没有 return 为 Transform<X, Y>,编译器会抛出错误。最好的方法是什么?

滥用运算符重载和折叠表达式 (C++17),您可能会做类似的事情来链接操作:

template <typename ... Fs>
class Chain
{
private: 
    template <typename F>
    struct CallerWrap
    {
        F f;
        CallerWrap(F f) : f(f) {}

        template <typename T>
        friend decltype(auto) operator+ (const CallerWrap& callerWrap, T&& x)
        {
            return callerWrap.f(std::forward<T>(x));
        }
    };

    std::tuple<Fs...> tup;  
public:

    Chain(Fs... fs) : tup{fs... }{}

    template <typename T>
    decltype(auto) operator()(T x) const {
        return std::apply([&x](auto... fs){ return (CallerWrap{fs} + ... + x); }, tup);
    }

};

Demo

f1(f2(f3(x))) 没有折叠表达式,但 f1 + (f2 + (f3 + x))(和其他二元运算符)有折叠表达式。 所以我们添加wrapper来使用后者,相当于前者。