Rust教程6:并发编程和线程通信
最佳答案 问答题库698位专家为你答疑解惑
文章目录
- 线程初步
- join方法
- 线程通信
Rust系列:初步⚙所有权⚙结构体和枚举类⚙函数进阶⚙泛型和特征
线程初步
在Rust中,开启多线程进行并发编程,只需调用thread::spawn,但这里有一个坑点,即spawn函数只有一个传入参数,即准备运行的函数,原则上讲这个准备并发执行的函数并没有输入参数。但从测试的角度来说,这样就没法分清谁是谁,为了让带有参数的函数也能享受多线程功能,可采取闭包的方式,将其转化为无参数函数,示例如下。
use std::thread;
use std::time::Duration;fn count(s:String){for i in 0..3{thread::sleep(Duration::from_millis(100));println!("{},{}", s, i);}
}fn main() {thread::spawn(|| -> count("A".to_string()));count("B".to_string())
}
其中,count的功能是,每隔100毫秒输出一个数字。在主函数中,调用了两次count,第一次开启了线程开关,第二次则在main的主线程中运行,结果如下,二者在输出上是没有先后顺序的,说明的确在并发执行。
B,0
A,0
B,1
A,1
A,2
B,2
join方法
如果讲上面代码的主函数改为调用两次spawn,像下面这样,那么运行之后,将不会有任何输出。原因也很简单,每一个线程开启之后,都会等待100毫秒才开始输出数字,但主线程却早早结束了。
fn main() {thread::spawn(||{count("A".to_string())});thread::spawn(||{count("B".to_string())});
}
为了避免这种尴尬的局面,可通过join方法,让主进程等待子进程跑完之后再继续,方法也很简单,只需写成下面的形式
fn main() {thread::spawn(||{count("A".to_string())}).join().unwrap();thread::spawn(||{count("B".to_string())}).join().unwrap();
}
输出结果如下,发现多线程貌似没起到作用。这个现象并不难理解,由于A线程后面紧跟着join,所以主线程会等待A线程执行完毕再继续。
A,0
A,1
A,2
B,0
B,1
B,2
为了避免这种尴尬的局面出现,可以将线程和join的位置分开,即给线程绑定一个变量,像下面这样,从而运行就正常了。
fn main() {let a = thread::spawn(||{count("A".to_string())});let b = thread::spawn(||{count("B".to_string())});a.join().unwrap();b.join().unwrap();
}
线程通信
在Rust中,线程通信需要引入另一个模块,mpsc(multiple producer, single consumer),使用其中的channel函数,生成一对可以穿透线程的电话
use std::thread;
use std::sync::mpsc;fn main() {let (tx, rx) = mpsc::channel();thread::spawn(move || {for i in 0..3{tx.send(i.to_string()).unwrap();}let val = String::from("hi");tx.send(val).unwrap();});loop{let received = rx.recv().unwrap();println!("Got: {}", received);if received=="hi"{return;}}
}
其中,tx在子线程中通过send进行发送,rx在主线程中通过recv()接收,运行结果如下
Got: 0
Got: 1
Got: 2
Got: hi
99%的人还看了
相似问题
猜你感兴趣
版权申明
本文"Rust教程6:并发编程和线程通信":http://eshow365.cn/6-36404-0.html 内容来自互联网,请自行判断内容的正确性。如有侵权请联系我们,立即删除!
- 上一篇: 四阶龙格库塔与元胞自动机
- 下一篇: golang正则获取中括号中的内容