rust 闭包与同步

rust 闭包与同步

rust 闭包

rust闭包中主要包括两个部分,闭包参数和闭包环境变量。
闭包函数和函数参数使用无太大区别。
闭包环境变量分为copy trait和非copy trait。
闭包分为move闭包和非move闭包。

copy trait变量,move闭包会将其单独移植进入闭包中,闭包内可修改,外部原始变量不受影响。
copy trait变量,非move闭包会将其原始变量移植入闭包,闭包内可修改,原始外部变量受影响。

非copy trait变量,move闭包会将其其单独移植进入闭包,闭包内可修改,外部原始变量失效。
非copy trait变量,move闭包会将其移入闭包,闭包内无法修改,外部原始变量不失效。

#[derive(Debug)]
pub struct Person {
    pub name: String,
    pub age: u32,
}

pub fn clone_struct_test_1() {
    let person_a = Person {
        name: String::from("wenbin"),
        age: 12,
    };

    let closure_fn = || person_a;

    let person_b = closure_fn();
    println!("{:?}", person_b);

    //    println!("{:?}",person_a);
}

pub fn clone_struct_test_1_1() {
    let mut person_a = Person {
        name: String::from("wenbin"),
        age: 12,
    };

    let closure_fn = || {
        let mut person_c = Person {
            name: person_a.name.clone(),
            age: person_a.age + 5,
        };
        //  this will report a error
        //        person_a.age = 17;
        person_c
    };

    let person_b = closure_fn();
    println!("{:?}", person_b);

    println!("{:?}", person_a);
}

pub fn clone_struct_test_1_2() {
    let mut person_a = Person {
        name: String::from("wenbin"),
        age: 12,
    };

    let closure_fn = move || {
        let mut person_c = Person {
            name: person_a.name.clone(),
            age: person_a.age + 5,
        };
        person_a.age = person_a.age + 12;
        (person_c, person_a)
    };

    let (person_b, person_a) = closure_fn();
    println!("{:?}", person_b);

    println!("{:?}", person_a);
}

pub fn clone_struct_test_2() {
    let mut person_a = Person {
        name: String::from("wenbin"),
        age: 12,
    };

    let closure_fn = |person_a| person_a;

    let mut person_b: &mut Person = closure_fn(&mut person_a);
    person_b.age = 13;
    println!("{:?}", person_b);

    println!("{:?}", person_a);
}

pub fn clone_struct_test_2_1() {
    let mut person_a = Person {
        name: String::from("wenbin"),
        age: 12,
    };

    let closure_fn = |person_a: &mut Person| {
        let mut person_c = Person {
            name: person_a.name.clone(),
            age: person_a.age + 23,
        };
        person_c
    };

    let mut person_b = closure_fn(&mut person_a);
    println!("{:?}", person_b);

    person_b.age = 13;
    println!("{:?}", person_b);
    println!("{:?}", person_a);
}

rust 同步

rust共享变量同步有mutex,RWLock,atomic。
mutex通常与arc结合,一起在多线程中使用。在多线程中使用需要将arc指针进行复制。
mutex的访问主要通过lock方法,当前线程阻塞时,会自动释放锁。

mutex自动通知,在rust中提供了根据event来通知其他线程的机制condvar。

// 代码中的Condvar就是条件变量,它提供了wait方法可以主动让当前线程等待,
// 同时提供了notify_one方法,让其他线程唤醒正在等待的线程。
// 这样就能完美实现顺序控制了。看起来好像条件变量把事都做完了,要Mutex干嘛呢?
// 为了防止多个线程同时执行条件变量的wait操作,
// 因为条件变量本身也是需要被保护的,这就是锁能做,而原子类型做不到的地方。
// 例子中可以看出,condvar和mutex想绑定,体现在wait方法:
// started = cvar.wait(started).unwrap();
pub fn sync_test_4(){
    let pair = Arc::new((Mutex::new(false), Condvar::new()));
    let pair2 = pair.clone();

    // 创建一个新线程
    thread::spawn(move|| {
        let &(ref lock, ref cvar) = &*pair2;
        println!("new thread start");
        let mut started = lock.lock().unwrap();
        *started = true;
        thread::sleep_ms(10000);
        cvar.notify_one();
        println!("notify main thread");
    });

    // 等待新线程先运行
    let &(ref lock, ref cvar) = &*pair;
    let mut started = lock.lock().unwrap();
    while !*started {
        println!("before wait");
        started = cvar.wait(started).unwrap();
        println!("after wait");
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。