在Rust中,只要参数是标识符(ident),就可以在宏参数中传递
是否可能创建一个宏来让您链接比较运算符?
>
或<=
等内容。是否可能创建一个宏来让您链接比较运算符?
let x = 3;
let y = 1;
let z = -3;
assert_eq!(cond!(z <= x > y), true);
可以的。你需要使用tt
来表示运算符类型:
macro_rules! cond {
(@rec ($head:expr) $last:ident $op:tt $next:ident $($tail:tt)*) => {
cond!(@rec (($head) && ($last $op $next)) $next $($tail)*)
};
(@rec ($head:expr) $last:ident) => { $head };
($first:ident $op:tt $next:ident $($tail:tt)*) => {
cond!(@rec ($first $op $next) $next $($tail)*)
}
}
fn main() {
let x = 3;
let y = 1;
let z = -3;
println!("(z <= x > y) = {}", cond!(z <= x > y));
}
您也可以阅读《Rust宏小书》了解更高级的宏模式。
我认为从技术上讲是可能的,但个人不确定是否要这样做。为了处理所有运算符,而又不匹配一些不应该工作的东西,如cond!(a + b)
、cond!(a)
或cond!()
,我必须非常详细,并使用递归宏。可能可以简化初始(非 @recur
)情况,但我担心做错会导致无限递归。
macro_rules! cond {
( @recur $x:ident ) => { true };
( @recur $x:ident < $y:ident $($tail:tt)* ) => { ($x < $y) && cond!( @recur $y $($tail)*) };
( @recur $x:ident > $y:ident $($tail:tt)* ) => { ($x > $y) && cond!( @recur $y $($tail)*) };
( @recur $x:ident <= $y:ident $($tail:tt)* ) => { ($x <= $y) && cond!( @recur $y $($tail)*) };
( @recur $x:ident >= $y:ident $($tail:tt)* ) => { ($x >= $y) && cond!( @recur $y $($tail)*) };
( @recur $x:ident == $y:ident $($tail:tt)* ) => { ($x == $y) && cond!( @recur $y $($tail)*) };
( $x:ident < $y:ident $($tail:tt)* ) => { ($x < $y) && cond!( @recur $y $($tail)*) };
( $x:ident > $y:ident $($tail:tt)* ) => { ($x > $y) && cond!( @recur $y $($tail)*) };
( $x:ident <= $y:ident $($tail:tt)* ) => { ($x <= $y) && cond!( @recur $y $($tail)*) };
( $x:ident >= $y:ident $($tail:tt)* ) => { ($x >= $y) && cond!( @recur $y $($tail)*) };
( $x:ident == $y:ident $($tail:tt)* ) => { ($x == $y) && cond!( @recur $y $($tail)*) };
}
fn main() {
let x = 3;
let y = 1;
let z = -3;
println!("(z <= x > y) = {}", cond!(z <= x > y));
}
cond
的参数上小心处理就可以了。tt
(对于参数operator0
)匹配<
、<=
、>=
等,以避免重复许多情况,但是tt
当然也会匹配其他标记。macro_rules! cond{
($x:ident $operator0:tt $x0:ident) => {
($x $operator0 $x0)
};
($x:ident $operator0:tt $x0:ident $($operator1:tt $x1:ident)*) => {
($x $operator0 $x0) && cond!($x0 $($operator1 $x1)*)
};
}
fn main() {
let x = 3;
let y = 1;
let z = -3;
assert_eq!(cond!(z <= x > y), true);
}
cond!(x+y)
。但它确实很简短、整洁。 - Michael Anderson