使用参数包的类似lisp的c++

lisp-like c++ using parameter packs

本文关键字:lisp c++ 参数      更新时间:2023-10-16

我对这个问题很感兴趣,所以我必须尝试一下,因为它让我想起了我绝对喜欢的lisp。这个问题中的例子:

auto list = [](auto ...xs) { 
    return [=](auto access) { return access(xs...); }; 
}; 
auto length = [](auto xs) { 
    return xs([](auto ...z) { return sizeof...(z); }); 
};
int main()
{
    std::cout << length(list(1, '2', "3")); // 3    
}

最终我得到了一些更多的多态lambdas,这进一步使它看起来像lisp:

auto l = list(-1, 2, 3, 4, 5, 6);
cons(unary::map(unary::inc)
            (binary::map(binary::add)
            (cdr(l), 1)), list(12, 13, 14, 15, 16))) 
// just an example, it doesn't really look that much like lisp, but still I was amused

现在,如果我想要一个打印函数,我必须这样写:

auto print = [](auto i)
{
    std::cout << i << " ";
    return i;
};

,然后将其映射到参数包中的每个参数。

那真的不像lisp,我想知道是否有一种方法可以递归地使用lisp中使用的car/cdr样式来遍历参数包。显然,必须使用list lambda返回的闭包。

另外,我知道参数包不应该真正用于这样的事情。

编辑:

我设法实现递归打印使用模板:

template<typename First>
void print_helper(First f)
{
    std::cout << f << std::endl;
}
template<typename First, typename... Rest>
void print_helper(First f, Rest... r)
{
    std::cout << f << " ";
    print_helper(r...);
}
template<typename Lambda>
void print(Lambda l)
{
    l([=](auto... elements)
    {
        print_helper(elements...);
    });
}

但是现在我有返回值的递归函数的问题。假设我想要有一个filter/remove-if函数,它返回满足所提供谓词给出的要求的元素列表。目前,我对每个元素使用std::pair(就像在链接的问题中一样),如果在打印时应该跳过对,则包含一个标志。有没有一种方法可以在不需要标记的情况下实际返回这些元素的列表?

我不太确定这是否是你想要的,但也许它有帮助:

住在这里
template<bool = true>
struct printer {
 template<typename List>
 static void run (List xs) {
  return xs([](
      auto first /* the car */,
      auto... rest /* the cdr, more or less */) {
   cout << first << " ";
   printer<(sizeof...(rest) > 0)>::run(
      list(rest...)); // building a new list every time, tough
  });
 }
};
template<>
struct printer<false> {
 template<typename List>
 static void run (List xs) {}
};
auto print = [](auto xs) { 
 return xs([=](auto ...z) {
  printer<(sizeof...(z) > 0)>::run(xs);
 });
};

虽然这只适用于编译时已知的列表,因为终止是通过模板形参完成的。


从你引用的问题来看,在顶部的答案中,最后有一个链接:https://gist.github.com/Manu343726/fb57746274fffc043c2a

这里的foldl_值得一看,它似乎完成了我必须使用帮助结构体的任务。


可以使用可变值,但必须在编译时知道列表的长度及其元素类型。