如何使用可变参数宏来调用嵌套构造函数?

sel*_*tze 6 macros rust

我正在尝试在Rust中创建一个允许我编写的宏

make_list!(1, 2, 3)
Run Code Online (Sandbox Code Playgroud)

代替

Node::new(1, Node::new(2, Node::new(3, None)))
Run Code Online (Sandbox Code Playgroud)

哪个应该适用于任意数量的"参数",包括零.这是我到目前为止:

macro_rules! make_list(
    () => (
        None
    );
        ( $x:expr, $( $more:expr ),* ) => (
        Node::new($x, make_list!( $( $more ),* ))
    )
);
Run Code Online (Sandbox Code Playgroud)

但是我收到以下错误:

error: unexpected end of macro invocation
  --> src/main.rs:19:42
   |
19 |             Node::new($x, make_list!( $( $more ),* ))
   |                                          ^^^^^
Run Code Online (Sandbox Code Playgroud)

我无法理解这一点.从我所知,它应该工作.我做错了什么?

完整的代码:

type List<T> = Option<Box<Node<T>>>;

struct Node<T> {
    value: T,
    tail: List<T>,
}

impl<T> Node<T> {
    fn new(val: T, tai: List<T>) -> List<T> {
        Some(Box::new(Node::<T> {
            value: val,
            tail: tai,
        }))
    }
}

macro_rules! make_list(
    () => (
        None
    );
    ( $x:expr, $( $more:expr ),* ) => (
        Node::new($x, make_list!( $( $more ),* ))
    )
);

fn main() {
    let _list: List<i32> = make_list!(1, 2, 3, 4, 5, 6, 7, 8, 9);
}
Run Code Online (Sandbox Code Playgroud)

Chr*_*gan 8

扩展错误:您可以了解只有一个值的情况,因此它会写入make_list!(1).但是,没有规则可以匹配,对于第二个规则,在使用表达式之后x,需要一个未提供的逗号.

所以你需要做到这一点,以便它可以工作,make_list!(1)而不仅仅是(事实上,不是)make_list!(1,).要实现此目的,请在重复部分内部使用逗号,如下所示:

macro_rules! make_list(
    () => (
        None
    );
    ( $x:expr $( , $more:expr )* ) => (
        Node::new($x, make_list!( $( $more ),* ))
    )
);
Run Code Online (Sandbox Code Playgroud)

额外奖励:你可以写,make_list![1, 2, 3]而不是make_list!(1, 2, 3)你想要的.

  • 这种方法不再有效;请参阅[此处](https://users.rust-lang.org/t/tail-recursive-macros/905/3)。 (2认同)