深入理解 Rust Syntax

Ownership Rules

  • Each value in Rust has a variable that’s called its owner.
  • There can only be one owner at a time.
  • When the owner goes out of scope, the value will be dropped.

内存申请 let s = String::from("xx");
内存释放: s 超出scop,变为不可用。rust自动添加drop调动代码, 归还内存

所有权规则:

堆、栈 中变量的其他 赋值方式:

  • stack-only: copy
    let x = 5;
    let y = x;  // x, y  都可用, 因为x 为栈上分配, 对于内存方式为 copy, 不影响所有权
    
  • Heap: clone、所有权转移
let s1 = String::from("hello");
let s2 = s1.clone();
// s2 copy s1的内存,s1 依然可用
let s1 = String::from("xx");
let s2 = s1; // s1 不在可用, s2为 字符串的 owner, 这里只是转移 指向 string的指针,而非 copy string

Function 调用:

  • Function 调用参数: 跟 赋值 一样的所有权 转移一样
  • Function 返回参数: 堆上的内存,作为返回值的时候, 两种情况:
    1. 转移所有权 到 函数调用者
    2. drop掉 (因为超过 函数中的作用域 scope)

引用: Function 调用: 每次都需要转移所有权,在将所有权转回到调用者。非常麻烦, 所以设计了 引用

指向变量的指针, 并不具有 ownership, 所以drop并不会,释放内存,使得引用的变量不可用。在Function 中使用 非常合适,因为不需要ownership传递回去, 因为根本没有ownership的转移

  • 可变 mut 引用: 可以 改变 引用指向的内容。
  • 不可变引用
  • 引用 规则:
    1. 任何时候,只有一个 可更改引用,或者 同时多个不可变应用
    2. 引用需要总是有效的。即: 引用的scope应该小于变量的scope
fn no_dangle() -> &String {
  let s = String::from("hello");
  &s
}
// 函数返回, s 会drop掉, 所以会造成空指针 null reference

错误的函数,s 在 函数中分配内存,但是只返回引用,引用变量作用于大于 指向的变量作用域

  • slice: 同refrence, 引用一个连续的collection,但是没有ownership。 这里用来防止,在同样的作用域使用 mut refrence, 或者 mut 调用 (因为不能同时存在 mut 引用,和 非mut引用。所以自动的添加一份检查)

Generic define & syntax

定义 函数参数签名(告诉编译器 参数类型), 形式如下:

fn largest<T>(list:&[T]) -> T {
}

struct Point<T> {
    x: T,
    y: T,
}
// x, y 是同一类型, 也可以写成不一样的类型
struct Point<T, U> {
    x: T,
    y: U,
}

impl<T> Point<T> {
  fn x(&self) -> &T {
      &self.x
  }
} 

这里为什么显得如此怪异的原因, 在于 我们可以写出 impl Point<String> 来定定制 T=String 时候特有的方法定义。所以我们需要写成如此 impl<T> 来区分于 impl Point<String> , 声明 T 代表是一个place holder

并不会牺牲性能,没有runtime的耗时, 在编译阶段, rust 会填充 placeholder, 来完成, 不同类型的定义。

Trait: Defining Shared Behavior

  • impl 定义 及其 实现
  pub trait Summary {
      fn summarize(&self)-> String;
      fn speak(&self) -> {
          self.summarize();
          println!("speak");
      }
  }

  pub struct NewsArticle {
  }

  impl Summary for NewsArticle {
      fn summarize(&self) -> String {
          println!("---------");
      }
  }
  • Trait 类似于Interface, 共享 行为(函数) 定义,还可以 实现 类似 模板调用的方法。
  • Trait 当 Function 参数
pub fn notify(item: impl Summary) {
    println!("---------");
}
// Trait Bound syntax
pub fn notify<T: Summary>(item: T) {
      println!("---------");
}
// multiple Trait Bound
pub fn notify<T: Summary + Display>(item: T) {
      println!("---------");
}
// use where
pub fn some_function<T, U>(one: T, two: U) {
    where T: Display + Clone,
          U: Clone + Debug
    println!("---------");
}
  • Trait 当做Function 的return type, 但是 存在一些限制: 主要有, 返回值不能是不同类型, 而只能是一个确定的类型 impl trait(例如{} 中 通过if else 返回一个完全不同类型,却实现了同样的Trait 的类型)
  • Trait with Generic 可以 约束 impl Generic 的 类型为实现了 Trait 的类型。
impl<T: Display + PartialOrd> Point<T> {
    fn only_some(&self) {
    }// 只有实现了 Display & PartialOrd trait的 Point<_> 类型,才会有 only_some 方法
}
  • Advanced Traits: Traits with placeholder
  pub trait Iterator {
    type Item;

     fn next(&mut self) -> Option<Self::Item>;
  }
  impl Iterator for Counter {
      type Item = u32;

      fn next(&mut self) -> Option<Self::Item> {
      }
  }

为什么不适用这样的实现呢?

pub trait Iterator<T> {
    fn next(&mut self) -> Option<T>;
}

原因在于 如果采用第二种实现, 我们需要 写成这样

impl Iterator<String> for Counter {

}
impl Iterator<i32> for Counter {

}

这里面存在多种实现方式。 更重要的是,我们在调用next时候,需要显式的指定 .next::<Iterator<String>> 来 指导 rust使用哪个Iterator<T> for Counter 的代码实现。所以第一种更可取
但是确实存在 Generic 与 Trait 结合的例子:


trait Add<RHS=Self> {
    type Output;

    fn add(self, rhs: RHS) -> Self::Output;
}

// 常用的声明可以如下:
struct Point {
    x: i32,
    y: i32,
}

impl Add for Point {
    type Output = Point;

    fn add(self, other: Point) -> Point {
        Point {
            x: self.x + other.x,
            y: self.y + other.y,
        }
    }
}
// 然而我们依然可以这样,指定 RHS Generic参数, 来实现, Millimeters + Meters 的函数调用实现, 只不过,placeholder 并没有作为返回值,所有,可以直接+ 而不需要显示的,指定 + 之后的返回数值类型
struct Millimeters(u32);
struct Meters(u32);

impl Add<Meters> for Millimeters {
    type Output = Millimeters;

    fn add(self, other: Meters) -> Millimeters {
        Millimeters(self.0 + (other.0 * 1000))
    }
}

多个Trait 出现同样函数名字的情况:


trait Pilot {
    fn fly(&self);
}

trait Wizard {
    fn fly(&self);
}

//两个 Trait 拥有同样的函数名称,不同的函数实现。那如何在调用时候,决策函数调用实体呢?
struct Human;

impl Pilot for Human {
    fn fly(&self) {
        println!("This is your captain speaking.");
    }
}

impl Wizard for Human {
    fn fly(&self) {
        println!("Up!");
    }
}

impl Human {
    fn fly(&self) {
        println!("*waving arms furiously*");
    }
}
// 下面,person.fly 默认调用 Human 自己的实现, 如果需要 显式的调用 Pilot::fly 则需要, 如下格式
fn main() {
    let person = Human;
    person.fly();
    Pilot::fly(&person);
    Wizard::fly(&person);
    person.fly();
}
// 因为其为实例方法, 存在&self, 如果不存在呢? 下面示例:

trait Animal {
    fn baby_name() -> String;
}

struct Dog;

impl Dog {
    fn baby_name() -> String {
        String::from("Spot")
    }
}

impl Animal for Dog {
    fn baby_name() -> String {
        String::from("puppy")
    }
}

fn main() {
    println!("A baby dog is called a {}", Dog::baby_name());// Ok
    println!("A baby dog is called a {}", Animal::baby_name()); // Error, 下面是正解
    println!("A baby dog is called a {}", <Dog as Animal>::baby_name());
}


SuperTriat: 超级 Trait, 依赖于一个Trait的实现, 示例如下:


use std::fmt;
// 声明语法如下: trait SuperTrait: Trait {}
trait OutlinePrint: fmt::Display {
    fn outline_print(&self) {
        let output = self.to_string();
        println!("output is --- {}", output);
    }
}
// 我们只需要为point 实现 Display, 即可 拥有 outline_print 方法
impl fmt::Display for Point {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "({}, {})", self.x, self.y)
    }
}
impl OutlinePrint for Point {}

NewType: 因为 impl Trait for Type, 中的type需要在本地的crate,而不是 引用库 中的Type。 所以 可以通过Newtype类来包装 Type,实现一些 Trait.
这里面包含另外一些需要东西: 如何让 NewType, 伪装成Type?
实现 Deref Trait。

use std::fmt;

struct Wrapper(Vec<String>); // 新的type 类似于下边的
struct Color(i32, i32, i32);
struct Point(i32, i32, i32);

impl fmt::Display for Wrapper {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "[{}]", self.0.join(", "))
    }
}

fn main() {
    let w = Wrapper(vec![String::from("hello"), String::from("world")]);
    println!("w = {}", w);
}

Generic 生命周期 syntax: 用于区分函数中 参数生命周期, 对比 参数、返回值 等 生命周期之间的关系。 确保 参数传递生命周期符合 函数声明. 生命周期 需要关联 参数与返回值,才会有效果,只有参数的生命周期没有用处

fn longest(x: &str, y: &str) -> &str {
  if x.len() > y.len() {
      x
  } else {
      y
  }
}
// error, rust并不能知道 函数返回值, 是x还是y, 无法检查生命周期

fn longest<'a>(x: &'a str, y: &'a str) -> &'a str {
    if x.len() > y.len() {
        x
    } else {
        y
    }
}
// 要求 函数返回值,应该小于等于 参数, x y的生命周期, 所以下面的函数调用是可以pass的
fn main() {
    let string1 = String::from("long string is long");

    {
        let string2 = String::from("xyz");
        let result = longest(string1.as_str(), string2.as_str());
        println!("The longest string is {}", result);
    }
}

// 而这个, 则是编译失败的,因为 返回值的生命周期 大于其中参数 y 的生命周期, 会导致 dangling refrence, 比如, result指向 y, 而y 在 内部的scope中已经销毁了
fn main() {
    let string1 = String::from("long string is long");
    let result;
    {
        let string2 = String::from("xyz");
        result = longest(string1.as_str(), string2.as_str());
    }
    println!("The longest string is {}", result);
}
// 还可以这样, 总是 返回其中的一个值
fn longest<'a>(x: &'a str, y: &str) -> &'a str {
    x
}
  • Struct 的生命周期: struct 保持的refrence 的生命周期 与struct 生命周期关联。struct 不应该 长于 内部变量的refrence。
struct ImportantExcerpt<'a> {
  part: &'a str,
}

  fn main() {
      let novel = String::from("Call me Ishmael. Some years ago...");
      let first_sentence = novel.split('.')
          .next()
          .expect("Could not find a '.'");
      let i = ImportantExcerpt { part: first_sentence };
}// 其中 i 的生命从周期不应该长于 novel
// impl Struct Generic 方法时候的 声明语法, 同 impl Generic。 其中声明方法时候,需要不要 生命周期 声明,需要看,是否与struct field 、 返回值 相关
impl<'a> ImportantExcerpt<'a> {
   fn announce_and_return_part(&self, announcement: &str) -> &str {
       println!("Attention please: {}", announcement);
       self.part
   }
}

Static lifetime, 静态生命周期,表明变量, 将贯穿于整个program, 将直接保存于, 代码的二进制中。

Closures:

  • /|x/| {}
  • FnOnce: 获取参数的ownership
  • FnMut: 获取参数的 mut 引用
  • Fn: 获取参数的 非mut引用
  • 以上三种 为 Trait, 可以声明类型为 FN(i32) -> i32
  • function as paramas: Function Pointer(fn a Type diff with Fn) , fn 类型,实现了, Fn, FnMut, FnOnce 的实现, 即 impl Fn, FnMut, FnOnce for fn {....} 所以,可以传递 fn 到 接受 closures的 函数中。 还可以声明接受fn类型的 函数
 fn add_one(x: i32) -> i32 {
     x + 1
 }

 fn do_twice(f: fn(i32) -> i32, arg: i32) -> i32 {
     f(arg) + f(arg)
 }

 fn main() {
     let answer = do_twice(add_one, 5);

     println!("The answer is: {}", answer);
 }

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

推荐阅读更多精彩内容