结构体是用户定义的数据类型,其中包含定义特定实例的字段。结构有助于实现更容易理解的抽象概念。本文介绍几种初始化结构体对象的方法,包括常规方法、Default特征、第三方包实现以及构建器模式。
Struct声明与初始化
struct Employee {id: i32,name: String,address: String,designation: String,
}
每个字段的类型都写在它的名称前面(例如,id的类型是i32,名称是String,地址也是String,等等)。在声明结构体时,应给每个属性值赋,这个过程称为初始化。
struct Student {name:String,gpa:i32,faculty:String,
}fn main() {let std1 = Student {faculty:String::from("Computer Science"),name:String::from("Nil"),gpa:3};println!("Name is :{}, Faculty is :{}, GPA is :{}",std1.name,std1.faculty,std1.gpa);
}// 输出结果:Name is :Nil, Faculty is :Computer Science, GPA is :3
我们创建了一个名为Student的结构体,其中包含三个属性name、gpa和faculty。数据类型分别为String、i32和String。该结构在main()中初始化,然后利用println!宏打印结构的属性值。
实现Default 特征
实现Default trait可以为结构体提供默认值。我们可以通过对数据结构进行适度调整来使用自动生成的默认实现。当在数据结构中使用#[derived (Default)]时,编译器给每个属性中设置默认值。默认的布尔值为false,而默认的整数值为0。
#[derive(Default)]
struct Student {name:String,gpa:i32,faculty:String,
}fn main() {let std1 = Student {faculty: String::from("Computer Science"),name:String::from("Nil"),..Default::default()};println!("Name is :{}, Faculty is :{}, GPA is :{}",std1.name,std1.faculty,std1.gpa);
}// 输出结果:Name is :Nil, Faculty is :Computer Science, GPA is :0
gpa自动被设置了默认值,它的默认值是0。
使用 derivative 包
derivative包提供了一组可以自定义的#[derive]
属性。它可以帮助用户在结构体(struct)上自动实现一些复杂的行为。在初始化结构体属性方面,它可以结合属性宏(attribute macros)来提供灵活的初始化方式。需要使用cargo add derivative
引入依赖
use derivative::Derivative;use derivative::Derivative;
#[derive(Derivative, Debug)]
#[derivative(Default)]
struct Person {name: String,age: u32,
}#[derive(Derivative, Debug)]
#[derivative(Default)]
struct Student {#[derivative(Default(value = "-1"))]gpa:i32,
}fn main() {let person: Person = Person{..Default::default()};// println!("Student: {:?}", Person);// 此时person.name是一个空字符串,person.age是0println!("Student: {:?}", Student::default());
}
// 输出结果: Student: Student { gpa: -1 }
我们将gpa的默认值设置为-1, 初始化时不在需要给gpa传递任何值。
构建器模式
虽然derivative
本身没有直接提供构建器模式,但可以与构建器模式结合使用。构建器模式可以让用户更灵活地初始化结构体的属性,特别是当结构体有很多可选参数或者复杂的初始化逻辑时。
首先,定义一个PersonBuilder
结构体作为构建器:
use derivative::Derivative;
#[derive(Derivative)]
#[derivative(Default)]
struct Person {name: String,age: u32,
}
struct PersonBuilder {name: Option<String>,age: Option<u32>,
}
impl PersonBuilder {fn new() -> Self {PersonBuilder {name: None,age: None,}}fn name(mut self, name: String) -> Self {self.name = Some(name);self}fn age(mut self, age: u32) -> Self {self.age = Some(age);self}fn build(self) -> Person {Person {name: self.name.unwrap_or(String::from("Unknown")),age: self.age.unwrap_or(0),}}
}
下面是使用构建器方式初始化:
let person = PersonBuilder::new().name(String::from("Alice")).age(30).build();
更新方法
通过函数式更新初始化(在derivative
支持的派生 trait 基础上实现)。
假设我们已经有一个Person
结构体实例,并且想要创建一个新的实例,只修改其中的某些属性。首先,给Person
结构体实现一个update
方法(这可以通过derivative
来帮助派生一些辅助 trait,使得实现这个方法更方便):
use derivative::Derivative;
#[derive(Derivative)]
#[derivative(Default)]
struct Person {name: String,age: u32,
}
impl Person {fn update(mut self, name: Option<String>, age: Option<u32>) -> Self {if let Some(new_name) = name {self.name = new_name;}if let Some(new_age) = age {self.age = new_age;}self}
}
初始化测试代码:
let original_person = Person {name: String::from("Bob"),age: 25,
};
let new_person = original_person.update(Some(String::from("Charlie")), Some(35));
如果不希望更新age属性,直接传入None:
let emp = original_person.update(Some(String::from("Charlie")), None);
println!("emp:{:?}", emp);