Bootstrap

Rust基础

数据表示

变量

let i = 1

可变性与不可变性

  1. 默认不可变
  2. 使用mut修饰可变,或者再次声明赋值(又称为隐藏性)
fn main() {
    let mut i = 3;
    println!("{}", i);
    i =10 ;
    let i = 20;
    println!("{}", i);
}

常量

const MAX_POINT: u32 = 10000;

数据类型

  1. rust语言是静态类型语言,也就是说编译时必须知道所有变量的类型
  2. 编译器具有自动推导的能力

基础数据类型

bool

char 32 位 (其他语言大多是 8 位的)

数字类型

i8、i16、i32、i64、u8、u16、u32、u64、f32、f64

数组类型

let [Type;Size]

let x:[i32;4] = [1, 2, 3, 4];
println!("{:?}", x);

Size 也是数组类型的一部分

自适应类型(与操作系统位数有关)

isize、usize

复合数据类型

元组

let t1 = (1, 2, 3, 4);
let t2 = ("周", 1, '哈');
println!("{:?}", t1);
println!("{:?}", t2);
let (x,y,z) = t2;
println!("{}", x);

结构体

枚举

字符串类型

控制流

分支

fn main() {
    let age = 20;
    if age > 60 {
        println!("老年人")
    }else if age > 18 {
        println!("成年人");
    }else {
        println!("未成年人")
    }
}
let x = if true {5 } else { 6 };

循环

loop循环

fn main() {
    let mut i  = 0;
    let res = loop {
        if(i==10){
            break i;
        }
        i+=1;
        println!("in loop")
    };
    println!("{}", res);
}

while循环

fn main() {
    let mut i  = 0;
    while i != 10 {
        i+=1;
        println!("loop in");
    }

}

for循环

fn main() {
    let arr = [1, 2, 3, 4, 5];
    for element in arr {
        println!("{}", element);
    }
}

函数

函数参数类型不可自动推导

函数返回值类型不可自动推导

fn main() {
    println!("{:?}", other_fun1(1, 3));
}


fn other_fun(){
    println!("this is a function");
}

fn other_fun1(a:i32, b:i32) -> i32 {
   return a+b;
}
fn other_fun2(a:i32, b:i32) -> i32 {
    a+b
}

语句

fn main() {
    let i = 10;
    let x = {
        let y = 30;
        i + y
    };
    println!("{}", x);
}

所有权

  1. rust通过所有权机制来管理内存,编译器在编译就会根据所有权规则对内存的使用进行检查

  2. 堆和栈

    编译的时候数据的类型大小是固定的,就是分配在栈上的

    编译的时候数据 类型大小不固定,就是分配堆上的

  3. 作用域

    {}

    离开作用域调用drop方法

  4. String内存回收

  5. 移动

    fn main() {
        // 移动 始终只有一份有效
        let s1 = String::from("hello"); 
        let s2  = s1;
        println!("{}", s1);//s1 无效
    }
    
  6. clone

    fn main() {
        // 克隆 深度拷贝
        let s1 = String::from("hello");
        let s2  = s1.clone();
        println!("{}", s1);
    }
    
  7. 栈上数据拷贝(栈上移动)

    fn main() {
        let a = 1;
        let b = a;
        println!("{}", a);
        println!("{}", b);
    }
    

    copy trait

    只要实现copy 特质,可以继续使用

    具有copy特质的有哪些?

    ​ 所有的数字类型

    ​ 布尔类型

    ​ 字符类型

    ​ 元组

  8. 函数和作用域

fn main() {
    let i = 8;
    printNum(i);
    println!("{}", i);

    let str:String = String::from("helloworld");
    printString(str);
    println!("{}", str);
}



fn printString(str: String) {
    println!("{}", str);
}


fn printNum(num:i32) {
    println!("{}", num);
}

引用

用法 : &

让我们创建一个指向值的引用,但是并不拥有它,因为不拥有这个值,所以,当引用离开其指向的作用域后也不会被丢弃

借用

fn main() {
    let  mut str:String = String::from("helloworld");
    printString(&mut str);

    println!("{}", &str);
}



fn printString(str: &mut String) {
    str.push_str(" zhwenga");
    println!("{}", str);
}

悬垂引用 作用域内内存已被销毁,但返回了引用

fn main() {
    let refs = dangle();

}

fn dangle()-> &String{
    let string = String::from("hello");
    &string
}

在任意给定时间,有了可变引用之后不能再有不可变引用

slice

字符串slice是String中一部分值的引用

字面值就是slice

其他类型slice

fn main() {
    let s = String::from("hello world");
    let h = &s[0..5];
    let h = &s[..5];
    let h = &s[6..];
    let h = &s[6..=10];
    let h = &s[..];
    println!("{}", h);

    let arr = [1, 2, 3, 4];
    let x = &arr[1..3];
    println!("{:?}", x);
}

结构体

fn main() {
    // 1 定义结构体
    #[derive(Debug)]
    struct User{
        name:String,
        count:String,
        nonce:u64,
        active:bool
    }
    // 2 创建结构体实例
    let mut zhangsan = User {
        name: String::from("张三"),
        count: String::from("23442432"),
        nonce: 1000,
        active: true
    };
    // 3 修改结构体字段
    zhangsan.count = String::from("3424324242111");
    // 4 参数名字和字段同名的简化写法
    let name = String::from("李四");
    let count = String::from("34243224242");
    let nonce = 323434;
    let active = true;
    let lisi = User {
        name,
        count,
        nonce,
        active
    };
    println!("{:?}", lisi);
    // 5 从其他的结构体创建实例

    let wangwu = User {
        name: String::from("王五"),
        ..lisi
    };
    println!("{} {}", wangwu.name,wangwu.active);
    // 6 元组结构体(字段没有名字,圆括号)
    struct Point(i32,i32);
    let a = Point(10, 32);
    let b = Point(100, 320);
    let c = a.0 + b.1;
    println!("{}", c);
    // 7 单位结构体
    struct unit{}
    // 8 打印结构体

    println!("{:#?}", wangwu);
}

方法

fn main() {
    let dog = Dog {
        name: String::from("旺财"),
        weight: 32.0,
        height: 23.2
    };
    println!("{:?}", dog);
    println!("{}", dog.get_height());
}

#[derive(Debug)]
struct Dog{
    name:String,
    weight:f32,
    height:f32
}
impl Dog{
    fn get_name(&self) -> &str{
        &(self.name[..])
    }
    fn get_weight(&self) -> &f32{
        &(self.weight)
    }
}
impl Dog{
    fn get_height(&self) -> &f32{
        &(self.height)
    }
}

枚举

fn main() {
    // 1 类似与c语言的方式定义
    #[derive(Debug)]
    enum IpAddKind{
        V4,
        V6
    }
    #[derive(Debug)]
    struct IpAddr{
        kind:IpAddKind,
        address:String
    }
    let ipv4 = IpAddr {
        kind: IpAddKind::V4,
        address: String::from("192.168.0.0")
    };
    println!("{:?}", ipv4);
    // 2 rust语言提倡的方式定义
    // 3 可以是不同类型
    #[derive(Debug)]
    enum IpAddr2{
        V4(u8,u8,u8,u8),
        V6(String)
    }
    let ip2V4 = IpAddr2::V4(192,168,0,0);
    println!("{:?}", ip2V4);

    let ip2V6 = IpAddr2::V6(String::from("234,432,432,432,43"));
    println!("{:?}", ip2V6);

    // 4 经典用法
    #[derive(Debug)]
    enum Message{
        Quit,
        Move{x:i32,y: i32},
        Write(String),
        Change(i32,i32,i32)
    }

    // 5 枚举类型的方法以及match
    impl Message{
        fn prin(&self){
            match *self {
                Message::Quit => {println!("Quit")}
                Message::Move{x,y} => {println!("move {}{}",x,y)}
                Message::Write(_) => {println!("write")}
                Message::Change(_, _, _) => {println!("change")}
                _ => println!("Write")
            }
        }
    }
    let message = Message::Quit;
    message.prin();
    let moVe = Message::Move { x: 10, y: 20 };
    moVe.prin();

}

Option

fn main() {
    let some_number = Some(5);
    let some_string = Some(String::from("hello world"));
    let absent_number:Option<i32> = None;
    match some_number {
        None => {}
        Some(i) => {
            println!("{}", i);
        }
    }
}

vector

fn main() {
    // 1 创建空的vector
    let mut vec: Vec<i32> = Vec::new();
    // 2 创建包含初始值的vector
    let mut vec1 = vec![1, 2, 3];
    //3 丢弃vector
    // vec1.pop();

    // 4 读取元素
    // 5 遍历
    // 6 使用枚举
    vec1.push(32);
    for x in &vec1 {
        println!("{}", x);
    }
    match vec1.get(2) {
        None => {}
        Some(i) => {
            println!("{}", i);
        }
    };
    println!("{}", &vec1[0]);

    // 可变遍历
    for x in &vec1 {
        println!("{}", (*x + 2));
    }
}

string

fn main() {
    // 1 创建一个空String
    let mut str = String::new();
    str.push_str("hello");
    println!("{}", str);
    // 2 通过字面值创建一个String
    //     使用 String::from()
    //     使用 str
    let s1 = String::from("init some thing");
    println!("{}", s1);

    let mut s1 = "init some ".to_string();
    s1.push_str("heee");
    println!("{}", s1);
    // 3 更新 String
    let s1 = "hello".to_string();
    let s2 = String::from(",world");
    // let x = s1 + &s2;
    // println!("{}",x);
    let res = format!("{}{}", s1, s2);
    println!("{}", s1);
    println!("{}", s2);
    println!("{}", res);
    // 4 String 索引
    // 5 str 索引
    // 6 chars bytes
    let chars = "中国".chars();
    for x in chars {
        println!("{}", x);
    }
    for x in "万里长城".bytes() {
        println!("{}", x);
    }
}

hashmap

fn main() {
    // 1 HashMap
    // 2 创建HashMap
    // 3 读取
    // 4 遍历
    // 5 更新
    let mut map: HashMap<String,String> = HashMap::new();
    map.insert(String::from("name"),String::from("张三"));
    map.insert(String::from("name"),String::from("lisi"));
    // 不存在 则插入
    map.entry(String::from("one")).or_insert(String::from("32"));
    println!("{:?}", map);
    let name = map.get(&String::from("name"));
    match name {
        None => {}
        Some(name) => {
            println!("{}", name);
        }
    }
    for (key, value) in &map {
        println!("{} {}", key,value);
    }

}

包与模块

定义

  1. 包:Cargo的一个功能,允许构建、测试和分享crate
  2. Crate:一个模块的树形结构,形成库或二进制项目
  3. 模块:通过use来使用,用来控制作用域和路径的私有性
  4. 路径:一个命名例如结构体、函数或模块等项的方式

包和Crate

  1. crate root是一个源文件,Rust编译器以他为起始点,并构成您的crate的跟模块
  2. 包提供了一系列功能的一个或多个Crate
  3. Crate root是src/main.rs或者是src/lib.rs 说明:如果只有main.rs则说明这个包只有一个crate(main),如果同时拥有main。rs和其他的lib.rs(不一定是这个名字)则说明拥有多个crate
  4. crate会将一个作用域的相关功能分组到一起,使得该功能可以方便在多个项目之间共享

同一文件中

mod factory {
    pub mod produce_refrigerator {
        pub fn produce() {
            println!("生产冰箱")
        }
    }

    mod produce_washing {
        fn produce() {
            println!("生产洗衣机");
        }
    }
}

fn main() {
    factory::produce_refrigerator::produce();
}

导入外部库的使用

Cargo.toml

[package]
name = "rustPro"
version = "0.1.0"
edition = "2021"

# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html

[dependencies]
rust-crypto = "0.2"
use crypto::digest::Digest;
use crypto::sha3::Sha3;

fn main() {
    let mut hasher = Sha3::sha3_256();
    hasher.input_str("hello world");
    let resutl = hasher.result_str();
    println!("{}",resutl);
}

错误处理

fn main() {
    let f = File::open("hello.txt");
    let file1 = match f {
        Ok(file) => { file }
        Err(error) => { panic!("{}", error) }
    };
}
;