面试官:请问启动线程是start()仍是run()方法,能谈谈吗?java
应聘者:start()方法面试
当用start()开始一个线程后,线程就进入就绪状态,使线程所表明的虚拟处理机处于可运行状态,这意味着它能够由JVM调度并执行。可是这并不意味着线程就会当即运行。只有当cpu分配时间片时,这个线程得到时间片时,才开始执行run()方法。start()是方法,它调用run()方法.而run()方法是你必须重写的. run()方法中包含的是线程的主体(真正的逻辑)。 ide
继承Thread类的启动方式spa
public class ThreadTest { public static void main(String[] args) { MyThread t =new MyThread(); t.start(); } } class MyThread extends Thread{ @Override public void run() { System.out.println("Hello World!"); } }
实现Runnable接口的启动方式线程
public class ThreadTest { public static void main(String[] args) { Thread t =new Thread(new MyRunnable()); t.start(); } } class MyRunnable implements Runnable{ @Override public void run() { System.out.println("Hello World!"); } }
实际上这两种启动线程的方式原理是同样的。首先都是调用本地方法启动一个线程,其次是在这个线程里执行目标对象的run()方法。那么这个目标对象是什么呢?为了弄明白这个问题,咱们来看看Thread类的run()方法的实现:对象
public void run() { if (target != null) { target.run(); } }
当咱们采用实现Runnable接口的方式来实现线程的状况下,在调用new Thread(Runnable target)构造器时,将实现Runnable接口的类的实例设置成了线程要执行的主体所属的目标对象target,当线程启动时,这个实例的 run()方法就被执行了。blog
当咱们采用继承Thread的方式实现线程时,线程的这个run()方法被重写了,因此当线程启动时,执行的是这个对象自身的 run()方法。继承
总结起来:若是咱们采用的是继承Thread类的方式,那么这个target就是线程对象自身,若是咱们采用的是实现Runnable接口的方式,那么这个target就是实现了Runnable接口的类的实例。接口
咱们再来看一道混迹于各大面试公司笔试的题目:get
public class EqualsTest { public static void main(String args[]) { Thread t = new Thread() { public void run() { pong(); } }; t.run(); System.out.print("ping"); } static void pong() { System.out.print("pong"); } }
这里的标准答案是:pongping
这里直接调用线程的run方法,就至关于调用普通方法同样,由上往下执行,因此最后的结果是pongping。可是若是上面改为t.start()以后,这个结果就不固定了,由于这里有两个线程(其实还有一个守护线程,这里就先忽略),main线程和 t 线程,这两个线程得到cpu的时间就会不固定了,谁先得到CPU执行权,谁就先打印结果,因此最后的结果可能pongping也多是pingpong。