Rust 1.30带来更多元编程支持,并改进了模块系统

\u003cp\u003eRust的最新版本\u003ca href=\"https://blog.rust-lang.org/2018/10/25/Rust-1.30.0.html\"\u003e1.30\u003c/a\u003e扩展了过程宏,允许它们定义新的属性和类似于函数的宏。此外,它简化了Rust模块系统,使其更加一致、直观。\u003c/p\u003e\n\u003cp\u003eRust 1.30引入了两种新类型的\u003ca href=\"https://doc.rust-lang.org/book/first-edition/procedural-macros.html\"\u003e过程宏\u003c/a\u003e,“类属性的过程宏”和“类函数的过程宏”。过程宏是Rust元编程的基础,支持操作程序语法树。在这方面,过程宏要比\u003ca href=\"https://doc.rust-lang.org/reference/macros-by-example.html\"\u003e声明宏\u003c/a\u003e强大得多,声明宏提供了一种机制来定义基于模式匹配的更复杂代码的简写。\u003c/p\u003e\n\u003cp\u003e类属性过程宏类似于现有的派生宏,但是更灵活,因为它们允许你创建新的属性,并且除了结构和枚举之外,还可以应用于函数。例如,一个属性宏可以实现route属性规范,定义HTTP路由:\u003c/p\u003e\n\u003cpre\u003e\u003ccode\u003e// 使用路由过程宏\n#[route(GET, \u0026quot;/\u0026quot;)]\nfn index() {\n ...\n}\n\n// 过程宏定义路由\n#[proc_macro_attribute]\npub fn route(attr: TokenStream, item: TokenStream) -\u0026gt; TokenStream {\n // attr接收GET,宏的\u0026quot;/\u0026quot;部分\n // item接收fn index () { ...\n}\n\n\u003c/code\u003e\u003c/pre\u003e\n\u003cp\u003e类似地,类函数过程宏允许你定义类似于函数的宏,例如:\u003c/p\u003e\n\u003cpre\u003e\u003ccode\u003e// 解析SQL语句\nlet sql = sql!(SELECT * FROM posts WHERE id=1);\n\n#[proc_macro]\npub fn sql(input: TokenStream) -\u0026gt; TokenStream {\n\u003c/code\u003e\u003c/pre\u003e\n\u003cp\u003e在这两个示例中,TokenStream都表示应用属性的语法树或属性/函数定义。route/sql函数将接收到的语法树转换为返回给调用者的新语法树,即\u003ca href=\"https://tinkering.xyz/introduction-to-proc-macros/\"\u003e生成要执行的新代码\u003c/a\u003e。\u003c/p\u003e\n\u003cp\u003e在使用Rust模块系统时,Rust 1.30还对use宏进行了一些修改,以提升开发人员的体验。首先,use现在可以引入宏定义,从而淘汰了macro_use注解:\u003c/p\u003e\n\u003cpre\u003e\u003ccode\u003e// 旧:\n#[macro_use]\nextern crate serde_json;\n\n// 新:\nextern crate serde_json;\nuse serde_json::json;\n\u003c/code\u003e\u003c/pre\u003e\n\u003cp\u003e此外,通过把所有对名称空间的引用都与prelude模块中包含的所有extern crate指令进行比较,并使用匹配的那个,使得外部Crates对于在模块层次结构中移动的函数具有更强的适应性。以前,必须显式地在模块内部使用extern或者使用::extern_name语法,如下例所示:\u003c/p\u003e\n\u003cpre\u003e\u003ccode\u003eextern crate serde_json;\n\nfn main() {\n let json = serde_json::from_str(\u0026quot;...\u0026quot;); // OK\n}\n\nmod foo {\n\n // 为在模块种使用serde_json,必须显式使用use\n use serde_json;\n\n fn bar() {\n let json = serde_json::from_str(\u0026quot;...\u0026quot;);\n }\n\n fn baz() {\n // 也可以使用外部模块的完整限定名\n let json = ::serde_json::from_str(\u0026quot;...\u0026quot;);\n }\n\n\u003c/code\u003e\u003c/pre\u003e\n\u003cp\u003e最后,use现在以更加一致的方式解释模块路径。现在,你可以使用crate关键字来表明你希望模块路径从crate根路径开始。在1.30版本之前,这是默认的模块路径,但是,直接引用项的路径将从本地路径开始:\u003c/p\u003e\n\u003cpre\u003e\u003ccode\u003emod foo {\n pub fn bar() {\n // ...\n }\n}\n\nmod baz {\n pub fn qux() {\n \n // 旧\n ::foo::bar();\n // 无效,这和使用“use”不同:\n // foo::bar();\n\n // 新\n crate::foo::bar();\n }\n}\n\u003c/code\u003e\u003c/pre\u003e\n\u003cp\u003eRust 1.30还带来了如下变化:\u003c/p\u003e\n\u003cul\u003e\n\u003cli\u003e你现在可以通过加上前缀r#把关键字作为标识符,例如r#for。这种变化主要是因为Rust 2018将引入新的关键词,这样一种机制应当可以用于转换使用这些关键字作为变量或函数名的现有代码。\u003c/li\u003e\n\u003cli\u003e借助no_std,你现在可以构建应用程序而不使用标准库。以前,由于无法定义panic_handler,所以你只能使用no_std构建库。\u003c/li\u003e\n\u003c/ul\u003e\n\u003cp\u003e你可以使用$ rustup update stable更新Rust分发包。关于Rust 1.30的完整细节,请查阅\u003ca href=\"https://github.com/rust-lang/rust/blob/master/RELEASES.md#version-1300-2018-10-25\"\u003e发布说明\u003c/a\u003e。\u003c/p\u003e\n\u003cp\u003e查看英文原文:\u003ca href=\"https://www.infoq.com/news/2018/11/rust-1.30-released\"\u003eRust 1.30 Brings More Metaprogramming Support and Improved Modules\u003c/a\u003e\u003c/p\u003e\n

你可能感兴趣的:(Rust 1.30带来更多元编程支持,并改进了模块系统)