一.
结合上一节继续讲述,不要以为设置了标记线程就能停止,依旧有停不下来的情况。
整个函数就是添加了wait()方法,导致try-catch的加入。
DOS结果显示,程序没有停下来,和主线程结束了。其余两个线程都没结束,都处于等待状态。
这两个线程拿到锁进来后,都wait停在这了。
设置标记也没用,因为这两个线程就没读到标记。
有人说采用notify方法,但是notify要在同一个锁中,在这个程序中放不了。
如果程序中不用wait,用sleep呢?而且sleep时间比较长,比如说一个小时,没有一个小时根本醒不了。(这里就是说即使不是wait方法,换成sleep方法,也会出现线程没结束的情况,这样该怎么处理?)
接下来,介绍一种新方式。这种方式原理还是一样的,表现形式不一样。
什么叫中断?正在运行,突然停止了,这就是中断,中断其实就是一种冻结状态。而这个interrupt方法可以将中断状态清除掉。清除就是指它结束中断状态,继续运行。
interrupt是在将线程的冻结状态清除掉,让线程恢复到具备着cpu执行资格的状态。这和之前的notify有什么区别呢?不用notify,都能把线程从wait唤醒回来,不用等你时间到都可以从sleep状态回来。这种做法有点强制性,所以会抛出异常。它在不该醒的时候,你把它弄醒了。
我们将st.setFlag();该为两个线程中断。DOS结果显示如下,
(具体的内容再去看视频)
(但是)
如果你连标记都没读到,我就让你读标记,怎么让它读标记?直接调用interrupt先从状态中拉回来再说。中断异常,冻结状态异常。
当我们有些运行中的线程处在这种状态,但我们程序准备结束的时候,我们有可能强制地将这些线程从冻结状态拉回运行状态,让其结束。大家都知道,我们有三个线程,主线程结束了,它俩没结束,这个程序就结束不了。所以,必须要让所有的线程都结束,这个进程才能结束。不是所有线程都能结束的,系统的就结束不了。
package test5;
/**
*
* @author 高小硕
*但是如果线程处于冻结状态,无法获取标记,如何结束呢?
*
*可以使用interrupt()方法将线程从冻结状态强制恢复到运行状态中来,让线程具备cpu的执行资格。
*
*当时强制动作会发生了InterruptedException,记得处理。
*
*/
class StopThread implements Runnable
{
private boolean flag = true;
@Override
public synchronized void run()
{
while (flag)
{
try
{
wait();
}
catch (InterruptedException e)
{
// TODO: handle exception
System.out.println(Thread.currentThread().getName()+"......"+e);
flag = false;
}
System.out.println(Thread.currentThread().getName()+".......++");
}
}
public void setFlag() {
flag = false;
}
}
class StopThreadDemo {
public static void main(String[] args) {
// TODO Auto-generated method stub
StopThread st = new StopThread();
Thread t1 = new Thread(st);
Thread t2 = new Thread(st);
t1.start();
// t2.setDaemon(true);/守护线程
//t2.start();
t2.start();
int num = 1;
for (;;) {
if (++num==50)
{
// st.setFlag();
// t1.interrupt();
// t2.interrupt();
break;
}
System.out.println("main"+num);
}
System.out.println("Over");
}
}
参考后做了适当修改。http://www.cnblogs.com/wsw-bk/p/8097403.html