Java多线程面之线程生命周期与join方法

Java多线程面之线程生命周期与join方法

线程生命周期

如下图所示,文字性描述就不讲了。
在这里插入图片描述

代码案例

package com.thread.demo02;

public class MyThread extends Thread {

    private static int count = 0;

    public static void main(String[] args) throws InterruptedException {
        System.out.println(count);
        MyThread myThread = new MyThread();
        myThread.start();
        System.out.println(count);
    }

    @Override
    public void run() {
        for (int i = 0; i < 10000; i++) {
            MyThread.count = MyThread.count + 1;
        }
    }
}

此时,两次输出的count值都为0,第一个输出很简单,正常的main方法输出,此时count未做任何算数操作;第二个输出的情况是线程实例myThread调用start()方法,此时线程并未处于获取cup资源,运行run()方法,线程处于就绪未执行状态,count并未发生改变。

join()方法

java核心卷对join的描述:等待终止指定的线程。好像听不懂,其实就是某个线程调用了join方法,其主线程等待该线程执行完毕前处于阻塞等待状态。起作用是将线程串行化调用,如果每一个线程都调用join方法,则,每个线程根据线程实例调用join,依次执行线程。

package com.thread.demo02;

public class MyThread extends Thread {

    private static int count = 0;

    public static void main(String[] args) throws InterruptedException {
        MyThread myThread1 = new MyThread();
        myThread1.start();
        myThread1.join();
        System.out.println(count);
        MyThread myThread2 = new MyThread();
        myThread2.start();
        myThread2.join();
        System.out.println(count);
        MyThread myThread3 = new MyThread();
        myThread3.start();
        myThread3.join();
        System.out.println(count);
    }

    @Override
    public void run() {
        for (int i = 0; i < 10000; i++) {
            MyThread.count = MyThread.count + 1;
        }
    }
}

上面的线程每次执行结果,都是10000,20000,30000.线程串行化执行了。似乎没有任何问题,也没有线程安全问题,但是我们调整下代码试试。

package com.thread.demo02;

public class MyThread extends Thread {

    private static int count = 0;

    public static void main(String[] args) throws InterruptedException {
        MyThread myThread1 = new MyThread();
        myThread1.start();
        MyThread myThread2 = new MyThread();
        myThread2.start();
        MyThread myThread3 = new MyThread();
        myThread3.start();
        myThread1.join();
        System.out.println(count);
        myThread2.join();
        System.out.println(count);
        myThread3.join();
        System.out.println(count);
    }

    @Override
    public void run() {
        for (int i = 0; i < 10000; i++) {
            MyThread.count = MyThread.count + 1;
        }
    }
}

执行结果:

10000
22468
26980

好奇怪,这么会有线程安全问题产生了????答案是:调用join时,其他线程已经开始执行了。因此调用join不一定会保证线程的串行化执行,要在线程开启时,也就是调用join方法时,就开始调用join方法。(如果一定保证线程的串行化执行,最好在run()中,根据业务逻辑调用join方法是比较保险的。)
我们再次修改下代码:

package com.thread.demo02;

public class MyThread extends Thread {

    private static int count = 0;

    public static void main(String[] args) throws InterruptedException {
        MyThread myThread1 = new MyThread();
        myThread1.join();
        myThread1.start();
        System.out.println(count);
        MyThread myThread2 = new MyThread();
        myThread2.join();
        myThread2.start();
        System.out.println(count);
        MyThread myThread3 = new MyThread();
        myThread3.join();
        myThread3.start();
        System.out.println(count);
    }

    @Override
    public void run() {
        for (int i = 0; i < 10000; i++) {
            MyThread.count = MyThread.count + 1;
            //System.out.println(MyThread.currentThread().getName());
        }

    }
}

问题又来了,三个地方的输出总是0,0,0,在run方法中添加如下代码:

System.out.println(MyThread.currentThread().getName());

这里就不贴出输出结果了,(输出太多了)。根据输出结果发现三个线程是交替执行的。所以线程实例调用join没起作用,上面的线程周期图,可知,线程在执行start()方法时,线程处于就绪状态,可以抢夺cpu资源执行run()方法。但本代码是在start()之前调用join()方法,线程根本就没有开启。但为什么会输出三个0???通过输出每次输出线程的名称,可以发main已经获取cpu资源,执行完了三处输出操作,在执行三个并行的子线程。给人每次输出都是三个0的假象,如果执行的次数足够多的化,就不一定了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

狗大户中的狗大户

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值