2020 Rust 特征 (Trait)

rust.jpeg

选择 rust 的理由

  • Rust 有助于您提供代码质量
    • 让我们更加明确地了解性能成本
    • 便于开发人员权衡代码性能利弊
  • Rust 更加关注代码的质量和正确性
    • 强调内存安全,除非指定了"unsafe"
    • 强大的类型系统、匹配系统
  • 写 Rust 有一种写 kotlin 或者 go 这些高级语言的感觉

特征

组成我们应用通常是两个部分数据行为,我们编程多半工作就是用行为操作数据。

fn make_true(input:&str) -> String{
    format!("{}!!",input)
}

定义函数 make_true 对字符串进行操作,然后返回字符串,这是对数据的操作。

#[derive(Debug)]
struct Fact {
    text: String
}

fn make_true(input:&Fact) -> Fact{
    Fact{text:format!("{}!!",input.text)}
}

​往往将字符串定义在范围,我们将字符串作为对象 Fact 的属性,然后 make_true 接收 Fact 对象然后对其 text 属性进行操作。

我们希望方法与数据有一定关系,也就是方法属于数据,或者说想要将方法添加到数据上,在其他语言是 method。在 rust 为结构体添加行为很简单,

  • 通过关键字 impl 其后添加要添加方法到结构体名称
  • 然后写实现
impl Fact {
    fn make_true(&self) -> Fact{
        Fact{text:format!("{}!!",self.text)}
    }
}


fn main(){
   
    let fact = Fact{text:String::from("hello")};
    println!("{:#?}",fact.make_true());
}

上面代码大家疑问最多可能就是 self,为什么我们需要 self

  • 通过 self 方法可以访问到数据
  • 所以结构体内实现方法,都会得到数据引用作为方法第一个参数,然后来通过 self 引用来访问数据,类似 javascript 和 java 中的 this。
    根据需要可以指定不同 self
  • &self 借用、只读版本
  • &mut self 可变借用版本
  • [mut] self: 所有版本,可以操作修改 self
impl Fact {
    fn make_true(&mut self){
        self.text.push_str("!!");
    }
}
impl Fact {
    fn make_true(mut self) -> Fact{
        self.text.push_str("!!");
        self
    }
}

let fact = Fact{text:String::from("hello")};
let fact_1 = fact.make_true();
println!("{:#?}",fact_1.text);

特征定义

trait 用于定义与其他类型共享功能,这是一种抽象,类似于其他语言(例如 go 语言)中的接口。今天将介绍如何创建 trait 以及其实现和使用,最后会给出基于 trait 实现日志系统。

定义 trait

抽象方式定义共享的行为


pub trait GetInformation {
    fn get_title(&self)-> &String;
    fn get_course(&self)-> u32;
}

使用 trait 关键字来定义特征,然后在其中定义一系列行为,也就是空方法,需要结构体去实现。

pub struct Tut {
    pub title: String,
    pub course: u32,
}

impl GetInformation for Tut {
    fn get_title(&self) -> &String{
        &self.title
    }
    // u32 天然具有 copy 特征
    fn get_course(&self) -> u32{
        self.course
    }
}

定义结构体实现特征 GetInformation 的 get_title 这个结构体就具有 GetInfomation 特征。

fn main() {

    let js_tut = Tut{title:"vue".to_string(),course:10};
    println!("js_tut title = {} course = {}",js_tut.get_title(),js_tut.get_course())
}

在 go 语言我们可以根据行为进行划分类别,只要具有行为结构体就属于某一个按类别进行划分的类别。有时候我们传入结构体,需要具有一定能力。

实现 trait

作为参数输入结构体需要具有一定行为,也就是结构体实现某种特征

fn print_information(tut:impl GetInformation){
    println!("title = {}",tut.get_title());
    println!("age = {}",tut.get_course())
}


fn main() {

    let js_tut = Tut{title:"vue".to_string(),course:10};
    // println!("js_tut title = {} course = {}",js_tut.get_title(),js_tut.get_course())
    print_information(js_tut);

}

默认 Trait 实现

有些时候我们可以通过给出默认方法的实现,也就是给方法提供默认行为。

trait TutInfo {
    fn get_tut_info(&self) -> String{
        String::from("supplied by zidea zone")
    }
}

定义 TutInfo 特征,然后给行为定义默认行为,如果实现特征 TutInfo 没有复写 get_tut_info 方法时就会默认执行 TutInfo 特征默认提供的行为。

impl TutInfo for Tut {

    // add code here
}
let js_tut_info = js_tut.get_tut_info();
println!("tut info = {}",js_tut_info);

Trait 边界

trait 边界指定泛型是任何拥有特定行为的类型。

fn print_information_two<T:GetInformation>(tut:T){
    println!("title = {}",tut.get_title());
    println!("age = {}",tut.get_course());
}

这里就是使用特征边界(trait bound)来定义函数接受参数需要具有一定约束(要求结构体必须实现某种方法),所以约束就是要求结构体需要具有一定能力。

也可以指定多个特征边界(train_bound), 来约束对象具有多个行为。其实语法还是比较好理解,一门新语言带来很多新的特性,但是并不能改变你编程和设计程序能力。只是便于你对程序设计的实现。

trait GetTitle {
    fn get_title(&self) -> &String;
}

trait GetCourse {
    fn get_course(&self) -> u32;
}

impl GetTitle for Tut {
    fn get_title(&self)->&String{
        &self.title
    }
}

impl GetCourse for Tut {
    fn get_course(&self) -> u32{
        self.course
    }
}

fn print_information_three<T:GetTitle+GetCourse>(tut:T){
    println!("title = {}",tut.get_title());
    println!("age = {}",tut.get_course());
}

还有一种特征边界的写法,通过 where 对泛型进行限制,

fn print_information_five<T>(tut:T) where T:GetCourse + GetTitle{

    println!("title = {}",tut.get_title());
    println!("age = {}",tut.get_course());
}

接下来,也可以特征边界( trait bound) 来约束函数的返回值类型。

fn get_tut() -> impl GetTitle {
    Tut{
        title:String::from("react"),
        course:20
    }
}

这里需要补充一下这里 get_tut 返回值是 trait 类型,我们不能通过条件来返回不同都实现 GetTitle 的不同结构体,这样会报错

let reactTut = get_tut();
println!("title of react tut = {}",reactTut.get_title());

Trait 对象

在我们所熟悉的面向对象编程语言中,对象包含数据和行为。创建对象便可调用其方法(行为)进行操作。在 rust 我们将数据保存在 enums 或是 structs ,而行为写作 trait 里,也就是将数据和行为分开。Trait 对象行为更像传统的对象。Trait 对象可以包含数据和行为,但是又不同于传统对象。

  • 系统日志信息
  • 日志需要可配置
  • 在开发版提供详细信息,而在发布版提供简要的信息
    定义 Logger 结构体,添加输出不同级别日志信息
#[derive(Debug)]
struct Loggger {
}

impl Logger{
    fn error(&self, message:&str){}
    fn warn(&self, message:&str){}
    fn info(&self, message:&str){}
    fn debug(&self, message:&str){}
}

上面定义了不同级别的日志输出 error, warn, info 和 debug 级别日志输出

根据日志输出形式又定义不同类型日志输出

  • FilerLogger
  • PrintLogger
  • NullLogger
  • ExternalServiceLoagger
trait Loggger {
    fn error(&self, message:&str);
    fn warn(&self, message:&str);
    fn info(&self, message:&str);
    fn debug(&self, message:&str);
}

#[derive(Debug)]
struct PrintLogger {
}

impl Loggger for PrintLogger {

    fn error(&self, message:&str){
        println!("ERROR:{} ",message)
    }
    fn warn(&self, message:&str){
        println!("ERROR:{} ",message)
    }
    fn info(&self, message:&str){
        println!("ERROR:{} ",message)
    }
    fn debug(&self, message:&str){
        println!("ERROR:{} ",message)
    }
}

​这里我们定义特征 Logger,让不同日志输出类型都去实现 Logger 特征,这样他们就是不同类型却具有相同行为的类别

pub fn log_example(logger:&Logger){
    logger.info("runing example code ..");
    logger.info("done example code");
}

这里 &Logger 也称为Trait 对象,在 Trait 对象中包含

  • 一个指向一个基本类型(可能是 PrintLogger 也可能是 FileLogger)
  • 指针指向虚拟方法表(vtable)
    这些都是由编译器实现的


    Trait 对象

在 Trait 对象是包含一个指向堆上数据指针,以这种形式来保存数据,当堆上值大小变化不会影响到 Trait 对象,这样更利于分配内存给对象。

 logger.info("runing example code ..");

在上面语句编译器会先从 vtable 中加载 info 函数地址,然调用到 info 函数的地址。看整个过程,可能会担心效率,这样做会不会影响程序的性能。

vtable
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容