重庆网站建设公司 ,上海市杨浦区建设小学网站,公众号文章排版编辑器,免费发布产品信息网站文章目录前言1.start()方法和run()方法2.不能两次调用start()方法3.线程的执行是随机的start()方法和run()方法的执行顺序不一定相同4.run()方法由JVM调用public Thread(Runnable target)中的target前言
本篇文章主要讲解以下几个问题:
start()方法和run()方法的区别与联系为…
文章目录前言1.start()方法和run()方法2.不能两次调用start()方法3.线程的执行是随机的start()方法和run()方法的执行顺序不一定相同4.run()方法由JVM调用public Thread(Runnable target)中的target前言
本篇文章主要讲解以下几个问题:
start()方法和run()方法的区别与联系为什么不能连续两次调用shart()方法run()方法中可能忽略的问题
1.start()方法和run()方法
我们在创建线程时,会重写run()方法.run()方法可以理解为线程要做的任务,但是直接调用run()方法,只是main线程也就是主线程去执行的,是没有新线程产生的 如果要想让线程去执行run()方法里面的代码,就需要让创建线程的对象去调用start()方法,shart()方法可以创建并启动线程,JVM调用run()方法后(后面会有介绍) 线程才会去执行
2.不能两次调用start()方法
同一个线程对象,只能调用一次start()方法,不能两次调用start()方法,调用两次的话,会抛出 IllegalThreadStateException 这个异常 示例如下:
public class Example6 {public static void main(String[] args) {Thread t1 new Thread(()-{System.out.println(1);});t1.start();t1.start();}
}为什么会这样?我们可以查看start()的源码 上面有一段话: A zero status value corresponds to state NEW 意思是 零状态值对应于状态“NEW” 线程运行时有六种状态,先熟悉一下,之后还会写文章进行详细介绍:
状态描述新建(NEW)表示线程已经创建好了,但是还没有调用start()方法就绪(RUNNABLE)表示线程可能在运行,也可能在就绪队列阻塞 (BLOCKED)表示线程处于等在锁的状态等待(WAITING)表示线程处于条件等待状态,当触发条件后会唤醒计时等待(TIME_WAIT)比WAITING多了个超时条件触发的机制终止(TERMINATED)表示线程执行结束
因此线程再调用start()方法之后,可能处于终止,或者其它非NEW状态,第二次调用的时候,相当于重新让线程运行一遍,从线程安全和线程本身的执行逻辑来看,都是不合理的,因此在调用start()方法之后,会对线程的状态进行一个判断,如果线程不是在NEW状态下,就会抛出异常
3.线程的执行是随机的
线程的执行是随机的,这也是个老生常谈的问腿了,究其原因还是因为线程的抢占式执行,谁先抢到 操作系统分配的CPU资源,谁先去执行
start()方法和run()方法的执行顺序不一定相同
示例1: 看一下代码及代码运行结果
class MyThread extends Thread{private int val;public MyThread(int val) {this.val val;}Overridepublic void run() {System.out.println(val);}
}
public class Example1 {public static void main(String[] args) {MyThread myThread1 new MyThread(1);MyThread myThread2 new MyThread(2);MyThread myThread3 new MyThread(3);MyThread myThread4 new MyThread(4);MyThread myThread5 new MyThread(5);myThread1.start();myThread2.start();myThread3.start();myThread4.start();myThread5.start();}
}由此我们可以看到虽然是myThread1先调用的start()方法,但是输出的结果却是在第二位,而myThread3后执行却在第一位 再次运行一次代码的执行结果虽然与上次不同,但仍然是随机的,当然也不是没有运行结果是1 2 3 4 5的可能
示例2: 看一下代码及代码运行结果
public class Example4 {public static void main(String[] args) {Thread t1 new Thread(()-{while (true){System.out.println(t1);}});Thread t2 new Thread(()-{while (true){System.out.println(t2);}});t1.start();t2.start();}
}这里我们也能看到,start()执行的顺序与run()方法执行的顺序无关,随机执行t1和t2的run()方法 虽然这里是while(true),但是线程之间是相互独立的,所以并不影响 因此可以得出结论:start()方法和run()方法的执行顺序不一定相同
4.run()方法由JVM调用
start()执行完后,创建的新线程不会立刻去执行run()方法, run()方法会先被JVM去调用,之后对应的线程才会去执行.
public Thread(Runnable target)中的target
之前再讲创建线程的5种方法,介绍了实现Runnable接口,创建线程的方法 但其实这里面有一个坑,那就是public Thread(Runnable target)中的target参数,来看一下run()方法的底层源码 如果这里面的target不为空,才会去执行target的run()方法.如果传一个null,就不会去执行,如图所示: 代码如下:
class MyRunnable implements Runnable{Overridepublic void run() {System.out.println(1);}
}
public class Example7 {public static void main(String[] args) {MyRunnable myRunnable1 new MyRunnable();MyRunnable myRunnable2 null;Thread t1 new Thread(myRunnable1);t1.start();Thread t2 new Thread(myRunnable2);t2.start();}
}因此要注意使用继承Runnable接口 创建线程的时候 要注意target不要为null 文章到这里去结束了,感谢观看! “希望你继续兴致盎然的与世界交手,一直走在充满鲜花的道路”