rust 面向对象

1、struct 成员变量与成员方法分离

eg1

// 结构体的【成员变量】
struct Foo {
  age: i32,
  name: String,
}

// 结构体的【成员方法】
impl Foo {
  fn show(&self) {
    println!("age = {:?}, name = {:?}", self.age, self.name);
  }
}

fn main()
{
  let f = Foo {age:99, name:String::from("xiong")};
  f.show();
}
➜  main make
rustc main.rs
./main
age = 99, name = "xiong"
➜  main

eg2

struct Circle {
  x: f64,
  y: f64,
  radius: f64,
}

 /**
  * impl关键字在struct、enum或者trait对象上实现方法调用语法
  * 关联函数 (associated function) 的第一个参数通常为self参数。
  * 
  * 有3种变体:
  * => self,允许实现者移动和修改对象,对应的闭包特性为FnOnce
  * => &self,既不允许实现者移动对象也不允许修改,对应的闭包特性为Fn
  * => &mut self,允许实现者修改对象但不允许移动,对应的闭包特性为FnMut
  * => 不含self参数的关联函数称为静态方法 (static method)
 */
impl Circle {
  /**
   * 相当于类方法 => Circle::new() 调用 
   */
  fn new(x: f64, y: f64, radius: f64) -> Circle {
    // 创建Circle对象,省去return
    Circle {
      x: x,
      y: y,
      radius: radius,
    }
  }

  /**
   * 相当于实例方法 => Circle对象.area()
   */
  fn area(&self) -> f64 {
    self.x * self.y * self.radius
  }
}

fn main() {
  let c = Circle::new(1.0, 2.0, 3.0); // 堆区申请空间来存储对象数据
  println!("{}", c.area());
}
➜  main make
rustc main.rs
./main
6
➜  main

2、接口与实现

1. 接口定义与实现

// 接口
trait HasArea {
  fn area(&self) -> f64;
}

// 具体类
struct Rect {
  width: f64,
  height: f64,
}

// 让【具体类】实现【接口】
impl HasArea for Rect {
  fn area(&self) -> f64 {
    (self.width * self.height) // 作为返回值 => 必须使用 () 括起来,并不能写 ; 
  }
}

fn main()
{
  let r = Rect {width:10.5, height:20.5};
  println!("area = {:?}", r.area());
}
➜  main make
rustc main.rs
./main
area = 215.25
➜  main

2. 函数参数针对接口类型

1. fn run< T: HasArea>(r : &T);

// 接口
trait HasArea {
  fn area(&self) -> f64;
}

// 具体类
struct Rect {
  width: f64,
  height: f64,
}

// 让【具体类】实现【接口】
impl HasArea for Rect {
  fn area(&self) -> f64 {
    (self.width * self.height) // 作为返回值 => 必须使用 () 括起来,并不能写 ; 
  }
}

// 全局函数 => 参数类型为接口类型
fn run<T: HasArea>(r : &T) {
  println!("area = {:?}", r.area());
}

fn main()
{
  let r = Rect {width:10.5, height:20.5};
  run(&r);
}
➜  main make
rustc main.rs
./main
area = 215.25
➜  main

2. fn < T>(r : &T) where T: HasArea;

// 接口
trait HasArea {
  fn area(&self) -> f64;
}

// 具体类
struct Rect {
  width: f64,
  height: f64,
}

// 让【具体类】实现【接口】
impl HasArea for Rect {
  fn area(&self) -> f64 {
    (self.width * self.height) // 作为返回值 => 必须使用 () 括起来,并不能写 ; 
  }
}

// 全局函数 => 参数类型为接口类型
// fn run<T: HasArea>(r : &T) {
//   println!("area = {:?}", r.area());
// }
fn run<T>(r : &T) where T: HasArea
{
  println!("area = {:?}", r.area());
}

fn main()
{
  let r = Rect {width:10.5, height:20.5};
  run(&r);
}
➜  main make
rustc main.rs
./main
area = 215.25
➜  main

3. where 使用

// 接口1
trait Man {
  fn name(&self) -> String;
}

// 接口2
trait Animal {
  fn name(&self) -> String;
}

// 接口3
trait Runnable {
  fn run(&self);
}

fn make_pair<T, U>(a: T, b: U) -> (T, U) 
  where T: Man, 
        U: Animal + Runnable // 模板参数U必须实现两个接口
{
  (a, b)
}

fn main()
{}

4. 使用系统接口

use std::fmt::Debug;

fn foo<T: Clone, K: Clone + Debug>(x: T, y: K) {
    x.clone();
    y.clone();
    println!("{:?}", y);
}

fn bar<T, K>(x: T, y: K)
    where T: Clone,
          K: Clone + Debug
{
    x.clone();
    y.clone();
    println!("{:?}", y);
}

3. 接口中可以带有方法默认实现

trait Foo {
  fn foo(&self);
  
  // default method
  fn bar(&self) { println!("We called bar."); }
}

struct Baz;

impl Foo for Baz {
  fn foo(&self) { println!("foo"); }
}

fn main()
{
  let b = Baz{};
  b.bar();
}
➜  main make
rustc main.rs
./main
We called bar.
➜  main

4. 接口的继承

trait Foo {
  fn foo(&self);
  
  // default method
  fn foobar(&self) { println!("Foo::foobar()"); }
}

// 接口继承
trait FooBar : Foo {
  fn foobar(&self);
}

struct Baz;

impl Foo for Baz {
  fn foo(&self) { println!("foo"); }
}

impl FooBar for Baz {
  fn foobar(&self) { println!("FooBar::foobar()"); }
}

fn main()
{
  let b = Baz{};
  
  // 调用不同接口的实现
  Foo::foobar(&b);
  FooBar::foobar(&b);
}
➜  main make
rustc main.rs
./main
Foo::foobar()
FooBar::foobar()
➜  main

5、运算符重载

1. 重载 + 操作符

use std::ops::Add;

#[derive(Debug)]
struct Point {
  x: i32,
  y: i32,
}

// 给struct重载+运算符
impl Add for Point {
  // 1. 确定模板参数的具体数据类型
  type Output = Point;

  // 2. +运算符重载函数实现
  fn add(self, other: Point) -> Point {
    Point { x: self.x + other.x, y: self.y + other.y }
  }
}

fn main() {
  let p1 = Point { x: 1, y: 0 };
  let p2 = Point { x: 2, y: 3 };
  let p3 = p1 + p2;
  println!("{:?}", p3);
}
➜  main make
rustc main.rs
./main
Point { x: 3, y: 3 }
➜  main

其他运算符重载套路相似。

2. 实现 Deref<Target=T>接口,则 *T 返回 &T类型的值

use std::ops::Deref;

// 实体类
struct Animal<T> {
  value: T, // 成员value
}

// 实现 Deref<Target=T>,*对象 => 返回自定义的值
impl<T> Deref for Animal<T> {
  // 1. 使用【预定义】模板参数Target指定deref()的返回值类型
  type Target = T;

  // 2. 实现deref()
  fn deref(&self) -> &T {
    return &self.value // 返回转换成的结果值
  }
}

fn main() {
  let anim = Animal { value: "monkey" };
  println!("{}", *anim == "monkey");
  println!("{}", *anim);
}
➜  main make
rustc main.rs
./main
true
monkey
➜  main

3. 实现 Deref<Target=T>接口,&U 的值自动强制转换为 &T 类型

use std::rc::Rc;

fn foo(s: &str) {
  // borrow a string for a second
}

fn main()
{
  // String implements Deref<Target=str>
  let owned = "Hello".to_string();

  // Rc 实现了 Deref<Target=T> 接口
  // => &Rc对象,返回 &T
  let counted = Rc::new(owned);

  // therefore, this works:
  foo(&counted);
}
➜  main make
rustc main.rs
./main
➜  main

4. &U 的多重嵌套

struct Foo;

impl Foo {
  fn foo(&self) { println!("Foo() ..."); }
}

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

推荐阅读更多精彩内容

  • Python 面向对象Python从设计之初就已经是一门面向对象的语言,正因为如此,在Python中创建一个类和对...
    顺毛阅读 4,205评论 4 16
  • 翠玉青山水中画,劈波斩浪游龙湖。清风自来抚满面,醉美江山回魂鼓。
    八斗才001阅读 172评论 0 0
  • “没有敌意的坚决,不含诱惑的深情”是精神分析自体心理学理论家科胡特的一个经典表述。他重在强调父母或早期照顾者的人格...
    暖阳_1332阅读 313评论 0 2
  • 与其悲伤 不如梦一场 醒来 换一份清醒头脑 不至混帐 就像雨 总是会如期而至 快快慢慢 却似箭 已离弦万丈 停泊着...
    馒头小米粥阅读 162评论 0 1
  • 9月份开始,娃正式步入幼儿园大班的学习,这意味着“幼小衔接”开始了。以前我对“幼小衔接”也是持反对意见,为什么要让...
    Super_Hellen阅读 297评论 0 1