This shows you the differences between two versions of the page.
Both sides previous revision Previous revision Next revision | Previous revision | ||
devlogs:7_5_2022 [2022/05/07 19:53] root |
devlogs:7_5_2022 [2023/10/19 15:25] (current) |
||
---|---|---|---|
Line 105: | Line 105: | ||
- There can be only one **OWNER** at a time | - There can be only one **OWNER** at a time | ||
- When the **OWNER** goes out of scope, the variable is dropped | - When the **OWNER** goes out of scope, the variable is dropped | ||
+ | |||
+ | == Variable Scope == | ||
+ | simple shit | ||
+ | <code rust> | ||
+ | { // s is not valid here or before | ||
+ | let s = " | ||
+ | //wow s your so cool cause we can use you for things here! | ||
+ | } //s has been dropped | ||
+ | </ | ||
+ | |||
+ | see, simple. | ||
+ | |||
+ | == freeing memeory == | ||
+ | When we request memory and are done with it. We must free it. Every allocate should be paired with one free of the memory. | ||
+ | |||
+ | Garbage collected languages scan for unused memory periodically and free it. | ||
+ | Big boi languages expect you to do it yourself. | ||
+ | |||
+ | **RUST DROPS THAT SHIT WHEN IT GOES OUT OF SCOPE** | ||
+ | |||
+ | == Move == | ||
+ | |||
+ | With simple types that are stored on the stack. A copy of the data is made when we try to bind it to another variable. **This is a deep copy** | ||
+ | <code rust> | ||
+ | let x =2; | ||
+ | let y=x; | ||
+ | println!(" | ||
+ | println!(" | ||
+ | </ | ||
+ | that will compile. | ||
+ | |||
+ | However with data stored on the Heap such as a string it would be to time consuming and expensive to run. So rust would just store the pointer to the data in the new variable. | ||
+ | \\ | ||
+ | [[https:// | ||
+ | |||
+ | But what if s1 goes out of scope first and the data gets dropped?????????? | ||
+ | |||
+ | <code rust> | ||
+ | let s1 = String:: | ||
+ | let s2 = s1; // s1 has gone out of scope | ||
+ | println!(" | ||
+ | println!(" | ||
+ | </ | ||
+ | That will not compile. | ||
+ | |||
+ | |||
+ | [[https:// | ||
+ | |||
+ | by default none of rust's functions will **deep copy** any heap data so you can just assume that it **moves** it. | ||
+ | |||
+ | == Clone == | ||
+ | |||
+ | If you did wanna **deep copy** heap data. You can clone it. Strings derive clone. | ||
+ | |||
+ | <code rust> | ||
+ | let s1 = String:: | ||
+ | let s2 = s1.clone(); | ||
+ | |||
+ | println!(" | ||
+ | //Copied from the rust book | ||
+ | </ | ||
+ | |||
+ | == Copy == | ||
+ | |||
+ | As with the simple int example above. Simple stack stored traits can be copied and used over again. Even structs that contain these copyable types can be copied. If a type inherits the **drop** trait it cannot be copied. | ||
+ | |||
+ | <wrap lo> things that can be copied : u32, | ||
+ | |||
+ | an (i32,i32) can be copied but a (i32, | ||
+ | |||
+ | == Ownership and Functions == | ||
+ | |||
+ | its simple | ||
+ | <code rust> | ||
+ | |||
+ | fn main() { | ||
+ | let s = String:: | ||
+ | |||
+ | takes_ownership(s); | ||
+ | // ... and so is no longer valid here | ||
+ | |||
+ | let x = 5; // x comes into scope | ||
+ | |||
+ | makes_copy(x); | ||
+ | // but i32 is Copy, so it's okay to still | ||
+ | // use x afterward | ||
+ | |||
+ | } // Here, x goes out of scope, then s. But because s's value was moved, nothing | ||
+ | // special happens. | ||
+ | |||
+ | fn takes_ownership(some_string: | ||
+ | println!(" | ||
+ | } // Here, some_string goes out of scope and `drop` is called. The backing | ||
+ | // memory is freed. | ||
+ | |||
+ | fn makes_copy(some_integer: | ||
+ | println!(" | ||
+ | } // Here, some_integer goes out of scope. Nothing special happens. | ||
+ | |||
+ | //Copied from the rust book | ||
+ | </ | ||
+ | |||
+ | Things that derive copy, get copied into a function. and droppable types get moved. | ||
+ | Returning stuff follows a similar rule. | ||
+ | |||
+ | <code rust> | ||
+ | fn main() { | ||
+ | let s1 = gives_ownership(); | ||
+ | // value into s1 | ||
+ | |||
+ | let s2 = String:: | ||
+ | |||
+ | let s3 = takes_and_gives_back(s2); | ||
+ | // takes_and_gives_back, | ||
+ | // moves its return value into s3 | ||
+ | } // Here, s3 goes out of scope and is dropped. s2 was moved, so nothing | ||
+ | // happens. s1 goes out of scope and is dropped. | ||
+ | |||
+ | fn gives_ownership() -> String { // gives_ownership will move its | ||
+ | // return value into the function | ||
+ | // that calls it | ||
+ | |||
+ | let some_string = String:: | ||
+ | |||
+ | some_string | ||
+ | // moves out to the calling | ||
+ | // function | ||
+ | } | ||
+ | |||
+ | // This function takes a String and returns one | ||
+ | fn takes_and_gives_back(a_string: | ||
+ | // scope | ||
+ | |||
+ | a_string | ||
+ | } | ||
+ | //Taken from the rust docs.. Again please read that. and not this... | ||
+ | </ | ||
+ | |||
+ | S1 gets ownership from the functions return. | ||
+ | |||
+ | S2 is moved into the function and then is returend back to s3 | ||