标签:输出 stream end 表示 pipe addclass 对象 任务 条件
多条线程之间有时需要数据交互,下面介绍五种线程间数据交互的方式,他们的使用场景各有不同。
PS:关于volatile的详细介绍请移步至:Java并发编程的艺术(三)——volatile
这两种方式都采用了同步机制实现多条线程间的数据通信。与其说是“通信”,倒不如说是“共享变量”来的恰当。当一个共享变量被volatile修饰 或 被同步块包裹后,他们的读写操作都会直接操作共享内存,从而各个线程都能看到共享变量最新的值,也就是实现了内存的可见性。
这种方式能“传递”变量。当需要传递一些公用的变量时就可以使用这种方式。如:传递boolean flag,用于表示状态、传递一个存储所有任务的队列等。
用这种方式实现线程的开关控制。
// 用于控制线程当前的执行状态
private volatile boolean running = false;
// 开启一条线程
Thread thread = new Thread(new Runnable(){
void run(){
// 开关
while(!running){
Thread.sleep(1000);
}
// 执行线程任务
doSometing();
}
}).start();
// 开始执行
public void start(){
running = true;
}
等待/通知机制的实现由Java完成,我们只需调用Object类的几个方法即可。
// 共享的状态变量
boolean flag = false;
// 线程1
Thread t1 = new Thread(new Runnable(){
public void run(){
while(!flag){
wait();
}
}
}).start();
// 线程2
Thread t2 = new Thread(new Runnable(){
public void run(){
flag = true;
notifyAll();
}
}).start();
上述例子thread1未加同步。当thread1执行到while那行后,判断其状态为true,此时若发生上下文切换,线程2开始执行,并一口气执行完了;此时flag已经是true,然而thread1继续执行,遇到wait后便进入等待态;但此时已经没有线程能唤醒它了,因此就一直等待下去。
为什么notify需要加锁?且必须和wait使用同一把锁?
首先,加锁是为了保证共享变量的内存可见性,让它发生修改后能直接写入共享内存,好让wait所处的线程立即看见。
其次,和wait使用同一把锁是为了确保wait、notify之间的互斥,即:同一时刻,只能有其中一条线程执行。
为什么必须使用同步块的锁对象调用wait函数?
首先,由于wait会释放锁,因此通过锁对象调用wait就是告诉wait释放哪个锁。
其次,告诉线程,你是在哪个锁对象上等待的,只有当该锁对象调用notify时你才能被唤醒。
为什么必须使用同步块的锁对象调用notify函数?
告诉notify,只唤醒在该锁对象上等待的线程。
等待/通知机制用于实现生产者和消费者模式。
synchronized(锁A){
flag = true;// 或者:list.add(xx);
锁A.notify();
}
synchronized(锁A){
// 不满足条件
while(!flag){ // 或者:list.isEmpty()
锁A.wait();
}
// doSometing……
}
在之前的生产者-消费者模式中,如果生产者没有发出通知,那么消费者将永远等待下去。为了避免这种情况,我们可以给消费者增加超时等待功能。该功能依托于wait(long)方法,只需在wait前的检查条件中增加超时标识位,实现如下:
public void get(long mills){
synchronized( list ){
// 不加超时功能
if ( mills <= 0 ) {
while( list.isEmpty() ){
list.wait();
}
}
// 添加超时功能
else {
boolean isTimeout = false;
while(list.isEmpty() && isTimeout){
list.wait(mills);
isTimeout = true;
}
// doSometing……
}
}
}
管道流用于在两个线程之间进行字节流或字符流的传递。
步骤如下:
1. 在一条线程中分别创建输入流和输出流;
2. 将输入流和输出流连接起来;
3. 将输入流和输出流分别传递给两条线程;
4. 调用read和write方法就可以实现线程间通信。
// 创建输入流与输出流对象
PipedWriter out = new PipedWriter();
PipedReader in = new PipedReader();
// 连接输入输出流
out.connect(in);
// 创建写线程
class WriteThread extends Thread{
private PipedWriter out;
public WriteThread(PipedWriter out){
this.out = out;
}
public void run(){
out.write("hello concurrent world!");
}
}
// 创建读线程
class ReaderThread extends Thread{
private PipedReader in;
public ReaderThread(PipedReader in){
this.in = in;
}
public void run(){
in.read();
}
}
//
public static void main(String[] args){
// 开启一条线程
Thread t = new Thread(new Runnable(){
public void run(){
// doSometing
}
}).start();
// 调用join,等待t线程执行完毕
try{
t.join();
}catch(InterruptedException e){
// 中断处理……
}
}
标签:输出 stream end 表示 pipe addclass 对象 任务 条件
原文地址:http://blog.csdn.net/u010425776/article/details/54341405