如何将空白/空参数传递给Rust中的宏?

时间:2017-01-17 19:37:49

标签: macros rust

在某些情况下,您可能希望将参数传递给宏,该宏可以是某个文本,也可以是 nothing (空格,就像没有写入任何内容一样)。

鉴于这个起点:

macro_rules! testme {
    ($var:ident, $code:block) => {
        for i in 0..10 {
            let $var = i;
            { $code }
            if $var > 5 {
                println!("over 5");
            }
        }
    }
}

fn main() {
    testme!(myvar, {
        println!("{}", myvar);
    });

}

我们可能希望var 可选是可变的,假设宏体比上面的例子大,最好不要复制整个宏。

macro_rules! testme {
    (private $var:ident, $code:block, $var_qual:tt) => {
        for i in 0..10 {
            // imagine this is a lot more code :)
            let $var_qual $var = i;
            { $code }
            if $var > 5 {
                println!("over 5");
            }
        }
    };
    (mut $var:ident, $code:block) => {
        testme!(private $var, $code, mut)
    };
/*
    ($var:ident, $code:block) => {
        testme!(private $var, $code, )
        //                          ^ how to pass in a blank argument?
    };
*/
}

fn main() {
    testme!(mut myvar_mut, {
        myvar_mut += 10;
        println!("{}", myvar_mut);
    });
/*
    testme!(myvar_immutable, {
        println!("{}", myvar_immutable);
    });
*/
}

据我所知,没有办法传递一个空参数,取消注释/**/注释以查看错误。

是否可以将一个空参数传递给一个宏来制作一个这样的例子?

1 个答案:

答案 0 :(得分:2)

据我所知,它无法传递空白/空参数。

然而,可以传入本地定义的宏,该宏可选地添加前缀。

工作示例:

macro_rules! testme {
    (private $var:ident, $code:block, $var_qual_macro:ident) => {
        for i in 0..10 {
            // imagine this is a lot more code :)
            let $var_qual_macro!($var) = i;
            { $code }
            if $var > 5 {
                println!("over 5");
            }
        }
    };
    (mut $var:ident, $code:block) => {
        macro_rules! var_qualifier { ($v:ident) => { mut $v } }
        testme!(private $var, $code, var_qualifier)
    };
    ($var:ident, $code:block) => {
        macro_rules! var_qualifier { ($v:ident) => { $v } }
        testme!(private $var, $code, var_qualifier)
    };
}

fn main() {
    testme!(mut myvar_mut, {
        myvar_mut += 10;
        println!("{}", myvar_mut);
    });
    testme!(myvar_immutable, {
        println!("{}", myvar_immutable);
    });
}

在嵌套这样的宏时要小心,宏的名称(在这种情况下为var_qualifier)不是在不同的宏中使用的相同名称,因为名称将被静默阴影化。