Rust 编程视频教程(进阶)——002 trait

视频地址

头条地址:https://www.ixigua.com/i6775861706447913485
B站地址:https://www.bilibili.com/video/av81202308/

讲解内容

1、trait用于定义与其它类型共享的功能,类似于其它语言中的接口。
(1)可以通过trait以抽象的方式定义共享的行为。
(2)可以使用trait bounds指定泛型是任何拥有特定行为的类型。

2、定义trait

pub trait GetInformation{
	fn get_name(&self) -> &String;
	fn get_age(&self) -> u32;
}

3、实现trait

//-----------------------------------------
pub struct Student {
	pub name: String,
	pub age: u32,
}

impl GetInformation for Student {
	fn get_name(&self) -> &String {
		&self.name
	}
	fn get_age(&self) -> u32 {
		self.age
	}
}
//----------------------------------------
pub struct Teacher {
	pub name: String,
	pub age: u32,
}
impl GetInformation for Teacher {
	fn get_name(&self) -> &String {
		&self.name
	}
	fn get_age(&self) -> u32 {
		self.age
	}
}

4、默认实现:可以在定义trait的时候提供默认的行为,trait的类型可以使用默认的行为。
例子:

trait SchoolName {
	fn get_school_name(&self) -> String {
		String::from("HongXingSchool")
	}
}

//下面使用默认实现
pub struct Teacher {
	pub name: String,
	pub age: u32,
}

impl SchoolName for Teacher {}
fn main() {
	let t = Teacher{ name: String::from("andy"), age: 32};
	let t_school_name = t.get_school_name();
	println!("teacher school name = {}", t_school_name);
}

5、trait作为参数
例子:

pub fn print_information(item: impl GetInformation) {
	println!("name = {}", item.get_name());
	println!("age = {}", item.get_age());
}

6、trait bound语法
上面的例子可以写成:

pub fn print_information<T: GetInformation>(item: T) {
	println!("name = {}", item.get_name());
	println!("age = {}", item.get_age());
}

7、通过+指定多个trait bound

pub trait GetName {
	fn get_name(&self) -> &String;
}

pub trait GetAge {
	fn get_age(&self) -> u32;
}

//写法一
pub fn print_information<T: GetName+GetAge>(item: T) {
	println!("name = {}", item.get_name());
	println!("age = {}", item.get_age());
}

//写法二,使用where
pub fn print_information<T>(item: T)
	where T: GetName+GetAge 
{
	println!("name = {}", item.get_name());
	println!("age = {}", item.get_age());
}

8、返回trait的类型

fn produce_item_with_information() -> impl GetInformation {
	Teacher {
		name: String::from("Andy"),
		age: 32,
	}
}

但是,需要注意的是,只能返回单一类型,以下实现会出错(因为返回了两个类型):

fn produce_item_with_information(is_teacher: bool) -> impl GetInformation {
	if is_teacher {
		Teacher {
			name: String::from("Andy"),
			age: 32,
		}
	} else {
		Student {
			name: String::from("harden"),
			age: 47,
		}
	}
}

9、复习之前的largest例子

fn largest<T:PartialOrd + Copy>(list: &[T]) -> T { //注意,要实现比较和复制的trait才行,否则报错
	let mut largest = list[0];
	for &item in list.iter() {
		if item > largest {
			largest = item;
		}
	}
	largest
}

fn main(){
	let number_list = vec![1, 2, 22, 3, 42];
	let r1 = largest_i32(&number_list);
	println!("r1 = {}", r1);
	let char_list = vec!['a', 'y', 'c', 'd'];
	let r2 = largest_char(&char_list);
	println!("r2 = {}", r2);
}

10、使用trait bound有条件地的实现方法

pub trait GetName {
	fn get_name(&self) -> &String;
}

pub trait GetAge {
	fn get_age(&self) -> u32;
}

struct PeopleMatchInformation<T, U> {
	master: T,
	employee: U,
}

impl <T: GetName+GetAge , U: GetName+GetAge> PeopleMatchInformation<T, U> {
	fn print_all_information(&self) {
		println!("teacher name = {}", self.master.get_name());
		println!("teacher age = {}", self.master.get_age());
		println!("student name = {}", self.employee.get_name());
		println!("student age = {}", self.employee.get_age());
	}
}

//使用
pub struct Teacher {
	pub name: String,
	pub age: u32,
}

impl GetName for Teacher {
	fn get_name(&self) -> &String {
		&(self.name)
	}
}

impl GetAge for Teacher {
	fn get_age(&self) -> u32{
		self.age
	}
}

pub struct Student{
	pub name: String,
	pub age: u32,
}

impl GetName for Student {
	fn get_name(&self) -> &String {
		&(self.name)
	}
}

impl GetAge for Student {
	fn get_age(&self) -> u32{
		self.age
	}
}

fn main() {
	let t = Teacher{ name: String::from("andy"), age: 32};
	let s = Student {name: String::from("harden"), age: 47};
	let m = PeopleMatchInformation{master: t, employee: s};
	m.print_all_information();
}

11、对任何实现了特定trait的类型有条件的实现trait
例如:

pub trait GetName {
	fn get_name(&self) -> &String;
}

pub trait PrintName {
	fn print_name(&self) ;
}

impl<T: GetName> PrintName for T {
	fn print_name(&self) {
		println!("name = {}", self.get_name());
	}
}

//----------使用----------------
pub struct Student{
	pub name: String,
}

impl GetName for Student {
	fn get_name(&self) -> &String {
		&(self.name)
	}
}

fn main() {
	let s = Student{name: String::from("Andy")};
	s.print_name(); //student实现了GetName trait,因此可是直接使用PrintName trait中的函数print_name
}
//引申:这种方式类似于c++里面的继承
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值