<style> .textleft { text-align:left; } .reveal, .reveal h1, .reveal h2, .reveal h3, .reveal h4, .reveal h5, .reveal h6 { font-family:Arial, Microsoft JhengHei;} .reveal .progress { height: 14px !important; } .progress span { background: url() repeat-x !important; } .progress span:after, .progress span.nyancat { content: ""; background: url() ; width: 34px !important; height: 21px !important; border: none !important; float: right; margin-top: -7px; margin-right: -10px; } .my-icon-css { width: 100%; height: 200px; background-position:center; background-repeat: no-repeat; background-image:url('https://mir-s3-cdn-cf.behance.net/project_modules/disp/fe36cc42774743.57ee5f329fae6.gif'); } </style> <!-- .slide: data-transition="slide" --> # <h2>ch5. Using Structs to Structure Related Data</h2> <div class="my-icon-css"></div> @Kais(VagrantPi) ###### tags: `Rust`, `簡報`, `Rust 讀書會` --- <!-- .slide: data-transition="convex" --> ## Defining and Instantiating Structs ---- <!-- .slide: data-transition="convex" --> ### Defining Structs ```go struct User { username: String, email: String, sign_in_count: u64, active: bool, } ``` ---- <!-- .slide: data-transition="convex" --> Structs 跟 tuples 很像,一樣可以包含多種 type,跟 tuples 不同的是每個 value 都有一個 key,所以可以清楚知道該值是啥 ---- <!-- .slide: data-transition="convex" --> ### Instantiating Structs - key: value pairs - 不須按造順序 ```go let user1 = User { email: String::from("someone@example.com"), username: String::from("someusername123"), active: true, sign_in_count: 1, }; ``` ---- <!-- .slide: data-transition="convex" --> #### 裡面有寫到一個不錯的比喻 > the struct definition is like a general template for the type, and instances fill in that template with particular data to create values of the type ---- <!-- .slide: data-transition="convex" --> #### 更改值 ```go let mut user1 = User { email: String::from("someone@example.com"), username: String::from("someusername123"), active: true, sign_in_count: 1, }; user1.email = String::from("anotheremail@example.com"); ``` ---- <!-- .slide: data-transition="convex" --> #### return Instance ```go fn build_user(email: String, username: String) -> User { User { email: email, username: username, // 因為輸入參數名稱跟 struct 變數名稱一樣,所以可以改寫成 // email, // username, active: true, sign_in_count: 1, } } ``` ---- <!-- .slide: data-transition="convex" --> ##### Creating Instances From Other Instances With Struct Update Syntax ``` 新的 Instance 有部分資料跟使用其他的 Instance 時, 可以透過 struct update syntax(..) 的方法將其餘沒寫到的變數資料補上 ``` ```go let user2 = User { email: String::from("another@example.com"), username: String::from("anotherusername567"), active: user1.active, sign_in_count: user1.sign_in_count, }; // -------------------------------------------------- let user2 = User { email: String::from("another@example.com"), username: String::from("anotherusername567"), ..user1 }; ``` ---- <!-- .slide: data-transition="convex" --> #### tuple structs ``` struct 跟 tuple 的混合型別,取值時是使用 index 值 ``` ```go struct Color(i32, i32, i32); struct Point(i32, i32, i32); let mut black = Color(0, 0, 0); let origin = Point(1, 2, 3); println!("{}", origin.1); // 2 black.0 = 1 println!("{}", black.0); // 1 ``` --- <!-- .slide: data-transition="convex" --> ## An Example Program Using Structs ---- <!-- .slide: data-transition="convex" --> v0.1 ```go fn main() { let width1 = 30; let height1 = 50; println!( "The area of the rectangle is {} square pixels.", area(width1, height1) ); // The area of the rectangle is 1500 square pixels. } fn area(width: u32, height: u32) -> u32 { width * height } ``` ---- <!-- .slide: data-transition="convex" --> <!-- 長跟寬是相關的,將他們包再一起比較好讀跟管理 --> v.02(使用 tuple 重構) ```go fn main() { let rect1 = (30, 50); println!( "The area of the rectangle is {} square pixels.", area(rect1) ); } fn area(dimensions: (u32, u32)) -> u32 { dimensions.0 * dimensions.1 } ``` ---- <!-- .slide: data-transition="convex" --> v.03(使用 struct 重構) ```go struct Rectangle { width: u32, height: u32, } fn main() { let rect1 = Rectangle { width: 30, height: 50 }; println!( "The area of the rectangle is {} square pixels.", area(&rect1) ); } fn area(rectangle: &Rectangle) -> u32 { rectangle.width * rectangle.height } ``` ---- <!-- .slide: data-transition="convex" --> #### Adding Useful Functionality with Derived Traits 如果是著想印出 struct ```go struct Rectangle { width: u32, height: u32, } fn main() { let rect1 = Rectangle { width: 30, height: 50 }; println!("rect1 is {}", rect1); // error[E0277]: the trait bound `Rectangle: std::fmt::Display` is not satisfied // `Rectangle` cannot be formatted with the default formatter; try using // `:?` instead if you are using a format string } ``` ---- <!-- .slide: data-transition="convex" --> println! 在印出東西時 {} 可以包含著格式化特徵(Formatting traits) 而 Display 為預設值 ```bash nothing ⇒ Display ? ⇒ Debug x? ⇒ Debug with lower-case hexadecimal integers X? ⇒ Debug with upper-case hexadecimal integers o ⇒ Octal x ⇒ LowerHex X ⇒ UpperHex p ⇒ Pointer b ⇒ Binary e ⇒ LowerExp E ⇒ UpperExp ``` ---- <!-- .slide: data-transition="convex" --> 之所以 Display 能印出原始 type,是因為都們很"單純",所以可以清楚的知道要印出什麼,但 struct 不同,包東包西,struct 的 `key` `value` 間還隔著 `:`,所以到底該印那些你並沒有說,而 `error` 也有建議可以使用 `:?` 來試試 ---- <!-- .slide: data-transition="convex" --> 當你試了之後會發現 ```go // error[E0277]: the trait bound `Rectangle: std::fmt::Debug` is not satisfied // `Rectangle` cannot be formatted using `:?`; if it is defined in your // crate, add `#[derive(Debug)]` or manually implement it ``` ---- <!-- .slide: data-transition="fade" --> ![](https://dvblobcdnjp.azureedge.net//Content/Upload/Popular/Images/2017-06/e99e6b5e-ca6c-4c19-87b7-dfd63db6381a_m.jpg) ---- <!-- .slide: data-transition="convex" --> #### 讓我們再印一次 ```go #[derive(Debug)] struct Rectangle { width: u32, height: u32, } fn main() { let rect1 = Rectangle { width: 30, height: 50 }; println!("rect1 is {:?}", rect1); // rect1 is Rectangle { width: 30, height: 50 } } ``` ---- <!-- .slide: data-transition="convex" --> #### 如果 struct 很大很複雜 這樣我們會改用 `{:#?}` ```go // rect1 is Rectangle { // width: 30, // height: 50 // } ``` --- <!-- .slide: data-transition="convex" --> ## Method Syntax ---- <!-- .slide: data-transition="convex" --> #### Defining Methods ```go #[derive(Debug)] struct Rectangle { width: u32, height: u32, } impl Rectangle { fn area(&self) -> u32 { self.width * self.height } } fn main() { let rect1 = Rectangle { width: 30, height: 50 }; println!( "The area of the rectangle is {} square pixels.", rect1.area() ); } ``` <!-- impl (implementation) --> ---- <!-- .slide: data-transition="convex" --> #### Methods with More Parameters ```go fn main() { let rect1 = Rectangle { width: 30, height: 50 }; let rect2 = Rectangle { width: 10, height: 40 }; let rect3 = Rectangle { width: 60, height: 45 }; println!("Can rect1 hold rect2? {}", rect1.can_hold(&rect2)); println!("Can rect1 hold rect3? {}", rect1.can_hold(&rect3)); // want // Can rect1 hold rect2? true // Can rect1 hold rect3? false } ``` ---- <!-- .slide: data-transition="convex" --> ```go impl Rectangle { fn area(&self) -> u32 { self.width * self.height } fn can_hold(&self, other: &Rectangle) -> bool { self.width > other.width && self.height > other.height } } ``` ---- <!-- .slide: data-transition="convex" --> #### Associated Functions <span>Associated Functions 通常用於建立一個或某種資料型態(因為會直接 return struct 的 instance)</span> ```go impl Rectangle { fn square(size: u32) -> Rectangle { Rectangle { width: size, height: size } } } let sq = Rectangle::square(3); ``` ---- <!-- .slide: data-transition="convex" --> #### Multiple impl Blocks ```go impl Rectangle { fn area(&self) -> u32 { self.width * self.height } } impl Rectangle { fn can_hold(&self, other: &Rectangle) -> bool { self.width > other.width && self.height > other.height } } ``` --- <!-- .slide: data-transition="convex" --> ## Thanks for listening ~
{"metaMigratedAt":"2023-06-14T16:53:55.622Z","metaMigratedFrom":"Content","title":"<h2>ch5. Using Structs to Structure Related Data</h2>","breaks":true,"contributors":"[{\"id\":\"69ade472-3ed3-499d-8a69-767243a31621\",\"add\":12228,\"del\":386}]"}
    1371 views