1.一般的代码
import java.util.concurrent.TimeUnit; public class TryConcurrency { private static void browseNews() { for ( ; ;) { System.out.println("uh-huh, the good news."); sleep(1); } } private static void enjoyMusic() { for ( ; ; ) { System.out.println("uh-huh, the nice music."); sleep(1); } } private static void sleep(int seconds) { try { TimeUnit.SECONDS.sleep(seconds); } catch (InterruptedException e) { e.printStackTrace(); } } public static void main(String[] args) { browseNews(); enjoyMusic(); } }
遗憾的是上面一直打印 uh-huh, the good news.
所以我们要引入Thread,重写其中的main方法
public static void main(String[] args) { new Thread() { public void run() { browseNews(); } }.start(); enjoyMusic(); }
start()方法是一个立即返回的方法,并不会让程序陷入阻塞
引入Lambda表达式
public static void main(String[] args) { new Thread(TryConcurrency::browseNews).start(); enjoyMusic(); }
会使代码更清晰
2.线程的生命周期
new(新建状态)
使用new创建了一个Thread对象时,并不处于执行状态,当.start()之后,就会进入runnable状态
runnable(可执行状态)
当调用run()方法之后,线程就进入了runnable状态,并且在jvm线程中创建了一个线程,但是此时线程仍然没有运行,它需要听从cpu的调度,称之为可执行状态
running(执行中状态)
当cpu在众多线程的可执行队列中选中了该线程,那么就会真正的执行自己的逻辑代码,该状态称之为运行状态
当线程调用stop()方法或者判断某个业务逻辑标识时,就会进入terminated状态
当线程调用sleep或者wait方法,就会进入blocked状态
当线程进行某个阻塞的io操作时,就会进入blocked状态
当获取某个资源文件而加入到阻塞队列中,就会进入blocked状态
当cpu的调度器轮询使该线程放弃执行,就会进入runnable状态
当线程调用yield方法,放弃cpu执行权,就会进入runnable状态
blocked(阻塞状态)
调用stop方法,或者意外死亡,就会进入terminated状态
线程阻塞结束,就会进入runnable状态
线程完成了指定休眠,就会进入runnable状态
等待状态(wait)线程被其他线程notify/notifyall唤醒,就会进入runnable状态
线程获取到某个锁的资源,进入runnable状态
线程在阻塞状态过程中被打断,比如其他线程调用了interrupt方法,线程就会进入runnable状态
terminated(死亡状态)
程序正常结束,线程意外死亡,jvm crash,线程进入terminated
3.我们在调用start方法时,为什么会执行run方法
是因为run方法被jni方法的start0调用,事实上threadStatus的内部属性是0;
不能两次启动Thread,否则会出现非法线程状态异常
线程启动后会被加入到ThreadGroup中
当线程进入死亡状态的时候,是不允许调用start方法,也就是说死亡状态下的线程是不允许回到可执行状态/执行状态
4.Thread的run和start就是一个典型的模板设计模式,父类编写算法结构代码,子类实现逻辑细节
5.Thread模拟营业大厅叫号机程序
待续
原文:https://www.cnblogs.com/freeht/p/12891268.html