【社区投稿】给Rust的Struct自动实现trait

这篇具有很好参考价值的文章主要介绍了【社区投稿】给Rust的Struct自动实现trait。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

给Rust的Struct自动实现trait

我们通常使用

#[derive(Clone, Debug)]

这样的方式给struct自动实现相应的trait,从而让struct具备某些特性,但是如果我们想让编译器给struct自动实现自己定义的trait要怎么办?

首先我们需要有一个trait,假设如下面的定义:

pub trait Printable {
    pub fn print_me(&self);
}

我们定义这个trait给struct赋予一个行为是逐行打印struct的所有Field。当然如果是自己实现肯定是可以凭空乱写的,那么我们可以和Debug一样,在 derive 中让编译器自动添加默认的实现。

首先需要给crate添加一个子crate:

cargo new --lib printable

然后在当前crate的 Cargo.toml 中添加依赖

[workspace]
members = [
    ".",
    "printable"
]
[dependencies]
printable = { version = "*", path = "printable"}

在printable 的 Cargo.toml 里还需要添加依赖

[dependencies]
syn = { version = "1.0", features = ["full"] }
quote = "1.0"
proc-macro2 = "1.0.51"

我们需要这三个crate来简化代码生成的工作,这里proc-macro2提供了自动实现宏的功能,syn用来解析结构体,quote用来输出TokenStream。

在 printable 的lib.rs 文件中

#[proc_macro_derive(Printable)]
pub fn print_info_derive(input: TokenStream) -> TokenStream {
}

我们在函数 print_info_derive 中输出的TokenStream,就会在编译时动态注入到struct中,这里参数input就是struct本身的代码流。

我们通过解析input就可以分析出 Struct的名字,Field列表,所有Field的名字,类型…..

下面是简化后的代码:

#[proc_macro_derive(Printable)]
pub fn print_info_derive(input: TokenStream) -> TokenStream {
    let struct_name = to_snake_case(input.ident.to_string().as_str());
        
    let fields = match input.data.clone() {
        syn::Data::Struct(data) => data.fields,
        _ => panic!("Only structs are supported"),
    };
    let fields_name: Vec<Ident> = fields.iter().map(|field| {
        field.ident.as_ref().unwrap().clone()
    }).collect();
}

之后我们就需要构建输出的代码流,这里使用 quote! 这个宏来实现。

#[proc_macro_derive(Printable)]
pub fn print_info_derive(input: TokenStream) -> TokenStream {
    let struct_name = to_snake_case(input.ident.to_string().as_str());
        
    let fields = match input.data.clone() {
        syn::Data::Struct(data) => data.fields,
        _ => panic!("Only structs are supported"),
    };
    let fields_name: Vec<Ident> = fields.iter().map(|field| {
        field.ident.as_ref().unwrap().clone()
    }).collect();
    let output_token = quote! {
        impl Printable for #struct_name {
            pub fn print_me(&self) {
                //这里添加逐行打印Field的代码,因为quote里本来就是在输出代码流
                //所以不能直接访问fields_name,比如循环之类的,所以我们这里需要
                //把生成这部分代码提取到函数外
            }
        }
    }
    output_token.into()
}

为了简单演示我们就使用一个函数来实现:

fn gen_print(fileds: Vec<Ident>) -> TokenStream2 {
    let print_stmts =fields.iter().map(|field| {
        quote! {
            println!("field:{}", &self.#field);
        }
    });
    quote!{
        #(#print_stmts)*
    }
}

最后组装一下,lib.rs  的代码如下:

#[proc_macro_derive(Printable)]
pub fn print_info_derive(input: TokenStream) -> TokenStream {
    let struct_name = to_snake_case(input.ident.to_string().as_str());
        
    let fields = match input.data.clone() {
        syn::Data::Struct(data) => data.fields,
        _ => panic!("Only structs are supported"),
    };
    let fields_name: Vec<Ident> = fields.iter().map(|field| {
        field.ident.as_ref().unwrap().clone()
    }).collect();
    let print_code = gen_print(fields_name);
    let output_token = quote! {
        impl Printable for #struct_name {
            pub fn print_me(&self) {
                #output_token
            }
        }
    }
    output_token.into()
}

如此这般一通操作后,我们随便一定一个Struct:

#[derive(Debug, Printable)]
struct TestTb {
    id: String,
    name: String,
    ts: i32
}

就可以

TestTb{id: "123".to_string(), name: "alex", ts: 111}.print_me();

就可以逐行打印出所有的Field了。

那么灵活的使用这个玩法,我们可以根据Struct的Field,自动生成 insert, update, delete的SQL也是可以的。给每个Field自动生成getter,setter方法…… (这个Java味太浓了,だめ)

研究这个是为了给 sqlx 增加一个自动生成insert,update,delete方法的增强,因为不喜欢写超长的insert和update语句。

Amusez-vous tous!文章来源地址https://www.toymoban.com/news/detail-828716.html

到了这里,关于【社区投稿】给Rust的Struct自动实现trait的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【Rust 基础篇】Rust Sized Trait:理解Sized Trait与动态大小类型

    Rust是一门以安全性和性能著称的系统级编程语言。在Rust中,类型大小的确定在编译期是非常重要的。然而,有些类型的大小在编译期是无法确定的,这就涉及到了Rust中的动态大小类型(DST)。为了保证在编译期可以确定类型的大小,Rust引入了Sized trait。本篇博客将深入探讨

    2024年02月14日
    浏览(38)
  • 【Rust 基础篇】Rust Deref Trait 的使用

    在 Rust 中,Deref trait 是一种特殊的 trait,用于重载解引用操作符 * 。通过实现 Deref trait,我们可以定义类型的解引用行为,使其在使用 * 运算符时表现得像引用类型。 本篇博客将详细介绍 Rust 中如何实现和使用 Deref trait,以及它在代码中的应用场景。 Deref trait 的定义如下:

    2024年02月17日
    浏览(38)
  • 研读Rust圣经解析——Rust learn-16(高级trait,宏)

    我们使用type即可声明一个关联类型,关联类型的作用就是简化和隐藏显示类型(个人认为) 简化:一个很长的类型总是被需要时,需要开发者耗费精力的重复书写,而且若有改动,则需要改多个地方 隐藏:对外部调用者隐藏,外部调用者无需知道它指的是什么,只要

    2024年02月02日
    浏览(67)
  • Rust-trait

    Rust语言中的trait是非常重要的概念。 在Rust中,trait这一个概念承担了多种职责。在中文里,trait可以翻译为“特征”“特点”“特性”等。 trait中可以定义函数。用例子来说明,我们定义如下的trait: 上面这个trait包含了一个方法,这个方法只有一个参数,这个self参数是什么意

    2024年01月20日
    浏览(43)
  • Rust语法: 枚举,泛型,trait

    这是我学习Rust的笔记,本文适合于有一定高级语言基础的开发者看不适合刚入门编程的人,对于一些概念像枚举,泛型等,不会再做解释,只写在Rust中怎么用。 枚举的定义与赋值 枚举的定义格式如下: enum 枚举名{ 值1(附加类型), 值2(附加类型),… } 其中,关联类型可以省去

    2024年02月13日
    浏览(39)
  • rust学习-泛型和trait

    Option,Vec,HashMapK, V,ResultT, E等,取函数以减少代码重复的机制 两个函数,不同点只是名称和签名类型 重写如下 为所有类型的结构体提供方法 只为f32提供方法 方法使用了与结构体定义中不同类型的泛型 Rust 实现了泛型,使得使用泛型类型参数的代码相比使用具体类型并没

    2024年02月17日
    浏览(43)
  • 30天拿下Rust之Trait

    概述         在Rust中,Trait是一个核心概念,它允许我们定义类型应该具有的行为。Trait类似于其他语言中的接口,但Rust的Trait更为强大和灵活。它不仅定义了一组方法,还允许我们指定方法的默认实现、泛型约束和继承。通过Trait,我们可以定义一组方法的签名和关联类

    2024年03月17日
    浏览(37)
  • Rust之泛型、trait与生命周期

    泛型是具体类型或其他属性的抽象替代。在编写代码时,可以直接描述泛型的行为,或者它与其他泛型产生的联系,而无须知晓它在编译和运行代码时采用的具体类型。 们可以在声明函数签名或结构体等元素时使用泛型,并在随后搭配不同的具体类型来使用这些元素。 当使

    2024年02月13日
    浏览(39)
  • Rust系列(四) trait备忘录(持续更新)

    上一篇:Rust系列(三) 类型系统与trait 基于官方文档进行简单学习记录,保证所有示例是可运行的基本单元。测试 rust 程序除了使用官方的 playground 之外,还可以通过定义 [[example]] 来运行程序。 用于 不可变对象 的解引用操作,语法类似 *v 。 官方文档: https://doc.rust-lang.org

    2024年02月14日
    浏览(33)
  • 【Rust笔记】意译解构 Object Safety for trait

    借助【虚表 vtable 】对被调用成员函数【运行时·内存寻址】的作法允许系统编程语言 Rust 模仿出 OOP 高级计算机语言才具备的【专用·多态 Ad-hoc Polymorphism 】特性。 计算机高级语言中的“多态”术语是一个泛指。它通常可被细化为 基于继承关系的“ 子类 ·多态”  Subtype Po

    2024年02月15日
    浏览(38)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包