繁体   English   中英

是否可以使用 C++17 折叠表达式仅折叠部分包装?

[英]Is it possible to fold only part of the pack with C++17 fold expressions?

我试图弄清楚如何使用 C++17 折叠表达式仅折叠可变参数模板包的一部分。 假设我想创建一个编译时“分隔字符串”,如"str1_str2_str3_....."

使用这样的代码可以很容易地完成(只是一个例子):

std::string result;

template<class... Strings>
ConcatString(Strings&... strs)
{
    (ConcatString(strs), ...);
}

template <class String>
void ConcatString(String& str)
{
     result += str + "_"
}

然后我们可以这样执行

std::string s1 = "s1", s2 = "s2", s3 = "s3";
ConcatString(s1, s2, s3);
// result == "s1_s2_s3_"

如您所见,最后一个分隔符有问题。 如果没有运行时检查,有什么办法可以避免这个问题? 我可以想象的一种解决方案是仅折叠 (N - 1) 个 args 并“手动”连接最后一个。

您可以递归调用ConcatString并使用 constexpr if 以避免运行时检查

template<typename String, typename... Strings>
void ConcatString(String& str, Strings&... strs) {
    if constexpr(sizeof...(strs) == 0) {
        result += str;
    } else {
        result += str + "_";
        ConcatString(strs...);
    }
}

是否可以使用 C++17 折叠表达式仅折叠部分包装?

不,折叠表达式将折叠整个包。 但是,我们可以做一些技巧来实现我们所需要的。

在这里,您的Concat function 可以简化为使用单个二进制折叠。

template<class String, class... Strings>
std::string Concat(const std::string& delimiter, const String& str, const Strings&... strs)
{
     return (str + ... + (delimiter + strs));
}

用法:

int main()
{
    std::cout << Concat(",", "a") << std::endl;
    std::cout << Concat(",", "a", "b") << std::endl;
    std::cout << Concat(",", "a", "b", "c") << std::endl;
}

Output:

一个
一,乙
a,b,c

现场演示

这里的技巧是我们将参数包拆分为一个单数“头”( str )和一个可变参数“尾”( strs )。 通过这种方式,我们让 function 参数列表将第一个元素从包中拉出。 很多C++11 风格的模板元编程都使用了这个技巧)。

另一种方法是为我们的参数包创建一组索引 0、1、...、N,然后对于我们的折叠逻辑,我们可以为第 0、第 N 甚至任意元素做一些特殊的事情。 您可以在漂亮的打印元组问题上找到这种方法的变体。 在 C++20 中,由于 C++20 中的模板 lambda,我们可以将所有逻辑移动到一个方法中,如下所示:

template<class... Strings>
std::string Concat(const std::string& delimiter, const Strings&... strs)
{
    return [&delimiter]<class Tup, size_t... I> (const Tup& tuple, std::index_sequence<I...>)
    {
        return (std::string{} + ... + (I == 0 ? std::get<I>(tuple) : delimiter + std::get<I>(tuple)));
    }(std::tie(strs...), std::make_index_sequence<sizeof...(strs)>{});
}

演示 2

这种丑陋的语法是我创建了一个 lambda 并在一个语句中调用它。 可以说,您可以改为通过std::invoke调用它使其更具可读性。

请注意,我们使用索引检查是否打印分隔符。

让我们使用索引检查技巧来只用分隔符连接彼此:

template<class... Strings>
std::string ConcatEveryOther(const std::string& delimiter, const Strings&... strs)
{
    return [&delimiter]<class Tup, size_t... I> (const Tup& tuple, std::index_sequence<I...>)
    {
        return (std::string{} + ... + (I % 2 == 0 ? std::get<I>(tuple) : delimiter + std::get<I>(tuple)));
    }(std::tie(strs...), std::make_index_sequence<sizeof...(strs)>{});
}

现在std::cout << ConcatEveryOther(",", "a", "b", "c", "d", "e", "f", "g") << std::endl; 会给我们一个 output 像

a,bc,de,fg

演示 3

怎么样:

template<typename Last>
std::string concat(Last last) {
    return last;
}

template<typename First, typename ...Rest>
std::string concat(First first, Rest ...rest) {
    return first + '_' + concat(rest...);
}

std::string s1{"foo"}, s2{"bar"}, s3{"baz"};
std::cout << concat(s1, s2, s3); // Prints "foo_bar_baz"

编辑:我使 function 更简单:

template <typename First, typename... Rest>
std::string concat(First first, Rest... rest) {
    return (first + ... + ('_' + rest));
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM