线程与进程
多线程,首先第一个会问的就是什么是线程?什么是进程?进程是代码在数据集合上的一次运行运动,是系统进行资源分配和调度的基本单位。线程是进程中的一个执行路径,一个进程中至少有一个线程(main),进程中的多个线程共享进程的资源。
关于操作系统的资源分配这里多说一句,操作系统是把资源分配给进程的,但是CPU资源比较特殊,它是直接分配给线程的。
线程的生命周期
1.start()函数
启动线程,线程由创建状态变为就绪状态,等待操作系统分配cpu执行片。
2.wait()函数
当线程调用一个共享变量的wait()方法时,该调用线程会被阻塞挂起,这个时候是会释放该共享变量的监视器锁,如果该线程还获得了其他共享变量的监视器锁,是不会释放其他变量的监视器锁。要调用一个共享变量的wait()方法必须先获取该共享变量的监视器锁,否则线程会抛出IllegalMonitorStateException异常。
wait方法返回的4种方式(1)其他线程调用了同一个共享变量的notify()方法或notifyAll()方法。(2)其他线程调用了该线程的interrupt()方法,该线程抛出InterruptedException并返回。(3)如果线程调用的是带超时时间的wait方法,则等待超时之后会返回。(4)虚假唤醒。为了避免虚假唤醒,应该多添加一个条件判断,如下伪码所示:
synchronizee(obj){
while(条件不满足){
obj.wait();
}
}
3.wait(long timeout)函数
和wait()的区别就是带了一个timeout时间,在指定的timeout ms时间内还没有被唤醒,该函数会因为超时而返回。
4.notify()函数
线程调用notify方法后,会唤醒一个在该共享变量上已调用wait方法的线程。一个共享变量上会有多个线程在等待,具体notify方法后唤醒的是哪个线程是随机的。
被唤醒的线程并不是马上从wait方法返回并继续执行,它必须在获取了共享变量的监视器锁后才可以返回并继续执行,也就是唤醒该线程的线程释放了共享变量的监视器锁之后,被唤醒的线程也不一定会获得共享变量的监视器锁,该线程需要和其他线程一起竞争监视器锁,获得了监视器锁后才可以继续执行。
5.notifyAll()函数
notify()方法是唤醒一个该共享变量上调用wait()方法而阻塞的线程,而notifyAll()方法是唤醒所有在该共享变量上阻塞的线程。notifyAll()方法只会唤醒在调用该方法前调用了同一个共享变量的wait()方法而阻塞的线程,也就是在notifyAll()调用之后的线程再调用同一个共享变量的wait()方法而阻塞的线程是不能够被唤醒的。
6.join()函数
join方法是在Thread类中,当一个线程A调用了线程B的join方法之后,线程A会被阻塞,直到线程B结束之后才会结束阻塞继续执行。
7.sleep(long millis)
指定的线程休眠millis毫秒之后再苏醒,调用sleep方法之后线程会让出cpu执行权,并在休眠的时间内不会再去参与cpu的调用,但是该线程获得的锁是不会释放的,其他线程即使在获得锁资源的线程休眠的情况下也是获得不到锁资源的。sleep方法正常返回之后,线程处于就绪状态,需要等待操作系统分配cpu时间片才能继续运行。
在调用Thread.sleep(long millis)millis参数传递一个负数,则会抛出IllegalArgumentExceptio异常。
8.yield()
调用yield()方法,是告诉线程调度器让出该线程获得的cpu执行权,进入到就绪状态,线程调度器会从线程就绪队列中获得一个线程优先级最高的线程,也有可能就是刚执行yield方法让出cpu执行权的线程。
9.线程中断 interrupt()、isInterrupted()、interrupted()
java中线程中断是一种线程间的协作模式,通过设置线程的中断标志并不是直接终止该线程的执行,而是被线程根据中断状态自行处理。
interrupt()方法:设置线程中断标志为true,并立即返回,线程只是被设置了中断标志,并没有被中断。如果线程A调用了wait方法、join方法或sleep方法从而阻塞挂起了,这时线程B调用了线程A的interrupt(),则线程A会在阻塞的地方抛出InterruptedException,并清除中断标识,这时调用isInterrupted方法返回的是false。
isInterrupted()方法:检测调用该方法的线程是否被中断。
interrupted()方法:检测当前线程是否被中断,是返回true,否则返回false。与isInterrupted()方法不同的是,interrupted()方法如果发现线程被中断,则会清除中断标志。
守护线程与用户线程
守护线程(daemon线程)和用户线程(user线程)区别之一是当最后一个非守护线程结束时,jvm会正常退出,而不管当前是否有守护线程,也就是守护线程是否结束并不影响jvm的退出。
默认创建的线程都是用户线程,设置线程为守护线程在调用线程start()方法之前调用thread.setDaemon(true)方法并设置为true即可。
main线程是一个用户线程,当main线程结束运行后,jvm会自动启动一个叫做DestoryJavaVM的线程,该线程会等待所有用户线程结束后终止jvm进程
今天的分享就到这,有看不明白的地方一定是我写的不够清楚,所有欢迎提任何问题以及改善方法。