—————-本系列作为学习张孝祥老师多线程教程的学习笔记——————-
来看这样一个题:子线程循环15次,接着主线程循环50次,接着又回到子线程循环15次,接着再回到主线程50次…如此循环100次,写出程序。
程序分析:首先我们将该程序分为2个部分,一是业务逻辑部分,用来做线程内最基本的循环;二是线程控制部分,来控制线程循环100次。
基于上面的思想,我们加一个业务逻辑类用来定义业务逻辑。
代码如下:
/**
*
* @author vayne 子线程执行15次,主线程执行50次,然后如此循环100次;
*/
public class TraditionalThreadCommunication
{
final static Business buse = new Business();
public static void main(String[] args)
{
new Thread(new Runnable()
{
@Override
public void run()
{
// TODO Auto-generated method stub
for(int i = 0;i<100;i++)
{
buse.sub(i);
}
}
}).start();
for(int i =0;i<100;i++)
{
buse.main(i);
}
}
}
// 业务逻辑类
class Business
{
public boolean bsub = false;//线程执行标示
//子线程的循环逻辑
public synchronized void sub(int i)
{
while (!bsub)
{
try
{
this.wait();
} catch (InterruptedException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
for (int j = 0; j < 15; j++)
{
System.out.println("sub sub sub" + j + ", loop" + i);
}
bsub = false;
this.notify();
}
//主线程的循环逻辑
public synchronized void main(int i)
{
while (bsub)
{
try
{
this.wait();
} catch (InterruptedException e)
{
// TODO Auto-generated catch block
e.printStackTrace();
}
}
for (int j = 0; j < 50; j++)
{
System.out.println("main" + j+", loop "+i);
}
bsub=true;
this.notify();
}
}
程序解读:
- bsub是一个标志,用来标识是否轮到该线程执行。
- while的使用:为什么用while而不用if呢?下面我们来解释一下,比如说,当bsub判断为true,主线程不能执行,当子线程执行完毕修改bsub标识之后,主线程会再次判断bsub,然后通知主线程开始执行。重点在于,通知之后,主线程的再次判断。其实线程是存在假醒状态的,即wait之后,没有等到通知自己就醒了。让我们来看一下文档:
上图截自JDK1.8的文档中Object的说明。如红线画的部分,“中断和假醒也是存在可能的,所以该方法应该放置在循环中使用”,那么没有等到通知就醒了会导致什么后果呢,如果假醒出现,那么if语句就不会再进行判断,就会继续往下执行,但是此时应该是属于另外一个线程执行的时间,这样就会打算原本的执行顺序,导致我们不期望的结果出现。