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:25] root |
devlogs:7_5_2022 [2023/10/19 15:25] (current) |
||
---|---|---|---|
Line 68: | Line 68: | ||
Think of a stack of plates: when you add more plates, you put them on top of the pile, and when you need a plate, you take one off the top. Adding or removing plates from the middle or bottom wouldn’t work as well! Adding data is called pushing onto the stack, and removing data is called popping off the stack | Think of a stack of plates: when you add more plates, you put them on top of the pile, and when you need a plate, you take one off the top. Adding or removing plates from the middle or bottom wouldn’t work as well! Adding data is called pushing onto the stack, and removing data is called popping off the stack | ||
- | < | + | < |
</ | </ | ||
+ | \\ | ||
+ | All data on the stack **MUST** have a knows fixed size at compile time | ||
+ | |||
+ | **Heap:** | ||
+ | The heap is less organized. You request space. The memory allocator finds you some thats big enough and gives you a pointer to it ((allocating memory)). Since the pointer is always of a fixed size, you can store that in the stack. | ||
+ | |||
+ | < | ||
+ | \\ | ||
+ | \\ | ||
+ | Think of being seated at a restaurant. When you enter, you state the number of people in your group, and the staff finds an empty table that fits everyone and leads you there. If someone in your group comes late, they can ask where you’ve been seated to find you. | ||
+ | |||
+ | < | ||
+ | </ | ||
+ | |||
+ | |||
+ | |||
+ | Pushing to the stack is faster <-> allocating to the heap is slower because it has to find memory rather than just //popping// it on top | ||
+ | |||
+ | Reading values from the stack is faster, the computer knows how much to jump to the next memory location <-> Reading form the heap is slower as the computer((NOT [[https:// | ||
+ | |||
+ | The computer would rather work on data on the stack as it is so much faster and cooler. | ||
+ | < | ||
+ | \\ | ||
+ | \\ | ||
+ | When your code calls a function, the values passed into the function (including, potentially, | ||
+ | < | ||
+ | </ | ||
+ | \\ | ||
+ | \\ | ||
+ | |||
+ | == Ownership Rules == | ||
+ | - Each value in Rust has a variable that is called its **OWNER** | ||
+ | - There can be only one **OWNER** at a time | ||
+ | - 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 | ||
+ | |||
+ | |||
+ | |||
+ | ~~DISCUSSION | Tell me the things I don't want to hear ~~ | ||