线程的实现方式:
① 继承Thread类(比较笨重,且因为Java只有单继承,拓展性不强,所以不推荐)
② 实现Runnable接口(推荐)
启动方式
Thead thread = new Thread(Runnable实现类);
thread.start(); // 执行的是Runnable实现类的run方法)(注:Thread类本身run方法没有任何动作)
对应的构造方法:
init()方法中将传入的Runnable接口(target)赋给了本类的field——target
run():
看见了吧,结合构造方法中执行的this.target = target知道,run执行的其实是传入的target的run方法。
而对于直接使用Thread,即Thead thread = new Thread(); thread.start(); target没有经过覆盖,此时target.run是什么呢?
再结合:
知,无参构造函数Thread()在进行初始化时,将本类的target置为了null,而run时如果发现本类target为null就不用运行run了,直接返回。
所以 —————— Thead thread = new Thread(); thread.start(); 时没有执行run()方法。
故有一个问题:如果传进去的是用户自定义的Runnable实现类,但是没有重写run方法,那么在start时不会将target覆盖(notNull),这样进入run方法时执行target.run()会发生什么——执行Runnable的run方法?
其实这是自己钻进去了,既然自己实现了Runnable接口,那么就得实现接口的全部方法——即run方法,不然连编译都无法通过,所以不存在上面假设的问题。
区分不同线程:
thread.setName("xxx");
thread.getName();
或者直接在创建线程时,指定名称,因为Thread提供这么一个构造方法:
PS:
① Thread是java.lang包内的,之所以能实现线程,因为它自己就实现了Runnable接口
② Runnable也是java.lang包内的,内容很简单,就一个 public abstract void run() :
故 Thread 本身就是一个线程的实现