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

视频地址

头条地址:https://www.ixigua.com/i677586170644791348...
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++里面的继承
本作品采用《CC 协议》,转载必须注明作者和本文链接
令狐一冲
讨论数量: 0
(= ̄ω ̄=)··· 暂无内容!

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!
文章
255
粉丝
119
喜欢
308
收藏
128
排名:335
访问:2.8 万
私信
所有博文
社区赞助商