线程
现在操作系统在运行程序时,会为其创建一个进程。现代操作系统调度最小单元是线程。一个进程里可以有多个线程。这些线程有自己独立的计数器堆栈局部变量等属性,并且能访问共享的内存变量。处理器来回在这些线程上切换,使用者感觉这些线程在同时执行。
Java天生就是多线程程序。main()方法是一个名为main的线程在运行。
为什么多线程
1、更多的处理器核心,使用多线程将计算逻辑分配到多个处理器核心,更有效率
2、更快的响应时间。多线程同时处理复杂计算,尽快完成逻辑,提升用户体验
3、更好的编程模型。Java为多线程编程提供了良好、考究并且一致的编程模型,使开发人员能够更加专注于问题的解决,即为所遇到的问题建立合适的模型,而不是绞尽脑汁地考虑如何将其多线程化。一旦开发人员建立好了模型,稍做修改总是能够方便地映射到Java提供的多线程编程模型上。
线程的状态
线程在运行的生命周期可能会有以下几种状态。在同一时刻只能处于其中一种状态
Daemon线程
Daemon线程是一种支持型线程。主要用作程序中后台调度以及支持性工作。如果JVM中不存在非Daemon线程,JVM可能会退出。可以通过Thread.setDaemon(true)将线程设置为Daemon线程。
启动终止线程
运行线程之前要先构造一个线程对象。一个新构造的线程对象是由其parent线程来进行空间分配的。child线程继承了parent是否为Daemon、优先级、加载资源的contextClassLoader以及可继承的ThreadLocal。同时还会分配一个唯一的id来标识这个child线程。此时线程状态为初始状态,在堆内存中等待运行。
启动线程
线程对象初始化完成以后,可以调用start()方法启动这个线程。线程start方法的含义是,当前线程同步告知JVM,只要线程规划器空闲,应该立即启动调用start方法。
注意:启动一个线程之前,最好为这个线程设置名称。
理解中断
中断可以理解为一个线程的标识位属性,他表示一个运行中的线程是否被其他线程进行了中断操作。中断好比其他线程对这个线程打了个招呼,其他线程调用该线程的interrupt方法对其进行了中断操作。
线程通过检查自身是否被中断来相应,线程通过方法isInterrupted()来进行判断是否被中断,也可以调用静态方法Thread.interrupted()对当前线程的中断标识位进行复位。如果该线程已处于终结状态,或者已经被中断过,在调用该线程对象的isInterrupted依旧会返回false。
安全的终止线程
除了中断以外,可以通过一个boolean变量来控制是否需要停止任务终止线程。
线程间通信
线程开始运行,拥有自己的栈空间,按照代码运行,直到结束。
volatile和synchronized关键字
Java多线程在同时访问一个对象或者对象的成员变量,每个线程都有这个变量的拷贝。在程序执行过程中,一个线程看到的变量不一定是最新的。
volatile可以修饰字段,告知程序任何对变量的访问需要从共享内存中获取,对它的改变必须同步刷新回共享内存,能保证所有线程对变量访问的可见性。
synchronized关键字修饰方法或者同步块,确保多个线程在同一个时刻,只有一个线程处于方法或者同步块中,保证了线程对变量访问的可见性和排他性。synchronized本质是对一个monitor进行获取,这个获取过程是排他的。只能有一个线程获取到,同一时刻。
任意一个对象都拥有自己的监视器。当这个对象由同步块或者这个对象的同步方法调用时,执行方法的线程必须先获取到该对象的监视器才能进入同步块或者同步方法,没有获取到监视器的线程会阻塞在同步块和同步方法的入口处,进入blocked状态。
等待通知机制
线程A调用了对象o的wait()方法进入等待状态,另一个线程B调用了O的notify()或者notifyAll()方法,线程A收到通知后从对象O的wait方法返回,进而进行后续操作。2个线程通过对象O来完成交互。对象上的wait和notify方法就像开关信号一样,用来完成等待方和通知方之间的交互工作
ThreadLocal的使用
threadlocal对象是以threadlocal对象为键,任意对象为值得存储结构。一个线程可以根据一个threadlocal对象查询到绑定在这个线程上的一个值。
可以通过set方法设置一个值,在当前线程在通过get方法获取到原先设置的值。