A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 史卜坤 中级黑马   /  2012-7-14 15:01  /  1194 人查看  /  2 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

本帖最后由 史卜坤 于 2012-7-14 15:03 编辑

class StopThread implements Runnable
{

        private boolean flag = true;

        public  synchronized void run()

        {

                while(flag)        

                {

                        try

                        {

                                wait();

                        }

                        catch (InterruptedException e)

                        {
                                System.out.println(Thread.currentThread().getName()+".....Exception");

                                //setFlag();


                        }

                        System.out.println(Thread.currentThread().getName()+".....");

                }

        }

        public void setFlag()

        {

                flag = false;

        }

}


class  StopThreadDemo

{

        public static void main(String[] args)

        {

                StopThread st = new StopThread();


                Thread t1 = new Thread(st);

                Thread t2 = new Thread(st);


                //t1.setDaemon(true);

                //t2.setDaemon(true);//将两个线程全标记成后台线程。


                t1.start();

                t2.start();


                int num = 1;

                while(true)

                {

                        if(num++==50)

                        {


                                 t1.interrupt();         // (1)

                                t2.interrupt();  // (2)

                                st.setFlag();        //  (3)


                                break;

                        }


                        System.out.println(Thread.currentThread().getName()+"....."+num);

                }

        }

}


这个是老师视频里的例题,我把(3)改动到1和2的下面,并把抛出异常中的setFlag();注释掉,按我的理解1,2,3是按顺序执行的,当程序执行到我注释的1,2的地方时,按我的理解是t1和t2线程结束冻结状态,抛出异常,打印System.out.println(Thread.currentThread().getName()+".....")语句后,这时又循环到了wait语句,再次冻结。这时运行3, 改变标记,但是 t1和t2还是在冻结冻结状态的啊,为什么程序能结束呢?

评分

参与人数 1技术分 +1 收起 理由
刘笑 + 1 赞一个!

查看全部评分

2 个回复

倒序浏览
1,2,3是按顺序执行的,
当执行到1,2时,t1和t2被打断,t1和t2会处理异常并执行下面的输出语句,
然后t1和t2会接着执行while循环,但是t1和t2是2个独立的线程,
从main线程的角度来看就是中断了t1和t2线程之后,马上就执行了setFlag
所以当t1和t2再一次回到while循环时,flag已经是false了
如果在main线程的setFlag前面加一个Thread.sleep(100);
就可以看见程序就不会结束了
回复 使用道具 举报
这个感觉有点绕,呵呵不过我也想看高手回答,但是一直没人,那我先说我的理解。
class StopThread implements Runnable
{
        private boolean flag = true;
        public  synchronized void run()
        {
        while(flag)        
                {
                        try
                        {                             
                                   System.out.println(Thread.currentThread().getName()+"-----waiting");                                wait();
                        }
                       catch (InterruptedException e)
                        {
                                System.out.println(Thread.currentThread().getName()
                                                +".....Exception");
                                //setFlag();
                        }
                        System.out.println(Thread.currentThread().getName()+".....");
                }
        }
        public void setFlag()
        {
                System.out.println(Thread.currentThread().getName()+ " is setFlag");               flag = false;
               System.out.println(Thread.currentThread().getName()+ " is runing");        }
}


class  StopThreadDemo
{
        public static void main(String[] args) throws InterruptedException
        {
               StopThread st = new StopThread();
               Thread t1 = new Thread(st);
               Thread t2 = new Thread(st);
               //t1.setDaemon(true);
                //t2.setDaemon(true);//将两个线程全标记成后台线程。
                t1.start();
                t2.start();
                int num = 1;
               while(true)
                {
                        if(num++==50)
                        {
                          t1.interrupt();
                         t1.sleep(1000);// (1)                           t2.interrupt();
                           t2.sleep(1000);// (2)                           st.setFlag();        //  (3)
                           break;
                        }
                     System.out.println(Thread.currentThread().getName()+"....."+num);
                }
        }
}
打印结果:Thread-0-----waiting
Thread-1-----waiting
main.....2
这两句不一定是在最前面
...
main.....50
Thread-0.....Exception
Thread-0.....
Thread-0-----waiting
Thread-1.....Exception
Thread-1.....
Thread-1-----waiting
main is setFlag
main is runing

添加红色代码,便于理解,也许添加的不合适,请高手帮忙修改。
开始运行后,t1,t2,都冻结,然后打印完异常抛出的语句后,又冻结了。这时主函数并没有结束。主函数去执行了setFlag。这样整个流程就结束了。

评分

参与人数 1技术分 +1 收起 理由
刘笑 + 1 赞一个!

查看全部评分

回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马