rust - 如何编写带有可链接标记的宏?

标签 rust

我不太确定如何表达这个问题,所以问题标题很垃圾,但这是我正在尝试做的事情:

我可以写这个宏:

macro_rules! op(
  ( $v1:ident && $v2:ident ) => { Op::And($v1, $v2) };
  ( $v1:ident || $v2:ident ) => { Op::Or($v1, $v2) };
);

我可以这样使用:

let _ = op!(Expr || Expr);
let _ = op!(Expr && Expr);

我想做的是像这样写一个任意序列的标记:

let _ = op!(Expr || Expr || Expr && Expr || Expr);

它解析为标记的 Vec,例如:

vec!(T::Expr(e1), T::Or, T::Expr(e2), T::Or, ...) 

我可以写一个vec!像宏:

macro_rules! query(
 ( $( $x:expr ),* ) => {
   {
     let mut temp_vec = Vec::new();
     $(temp_vec.push($x);)*
     temp_vec
    }
  };
);

...但我看不到如何在宏运行时将任意符号(例如 &&)转换为标记。

这有可能吗?

围栏链接:http://is.gd/I9F5YV

最佳答案

似乎不可能在宏扩展期间捕获任意符号匹配:正如语言引用所说,“有效的指示符是 item、block、stmt、pat、expr、ty(类型)、ident、path、tt”。所以我最好的建议是使用“ident”-有效的标记,比如“and”/“or”而不是“&&”/“||”,例如:

macro_rules! query_op(
  ( and ) => { "T::And" };
  ( or ) => { "T::Or" };
  ( $e:ident ) => { concat!("T::Expr(", stringify!($e), ")") };
);

macro_rules! query(
 ( $( $x:ident )* ) => {
   {
     let mut temp_vec = Vec::new();
     $(temp_vec.push(query_op!($x));)*
     temp_vec
    }
  };
);

fn main() {
  let q = query!(Expr1 or Expr2 and Expr3 or Expr4);
  println!("{:?}", q);
}

输出:

["T::Expr(Expr1)", "T::Or", "T::Expr(Expr2)", "T::And", "T::Expr(Expr3)", "T::Or", "T::Expr(Expr4)"]

关于rust - 如何编写带有可链接标记的宏?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28793406/

相关文章:

closures - 如何在嵌套映射中使用闭包?

windows - 如何告诉 rustc(通过 cargo)在哪里可以找到我的 dll 导入库

winapi - 使用Rust语言使用winapi SetClipboardData

rust - 如何解包 &Result<_,_>?

rust - Juniper 和 GraphQLEnum - Unresolved 导入 InputValue

rust - 是否可以限制 Criterion 执行的迭代次数?

objective-c - Rust bindgen clang 错误,此 __builtin_neon 函数的常量不兼容

unix - 我如何从 Rust 中的特定原始文件描述符中读取?

string - 切片字符串会复制底层数据吗?

rust - 为什么entry需要key的所有权?