java多线程
进程和线程的区别
进程是资源分配的最小单位,线程是CPU调度的最小单位所有与进程相关的资源,都被存在PCB中
进程是抢占处理机的调度单位,线程属于某个进程,共享其资源
线程只由堆栈寄存器/程序计数器/TCB组成
线程不能看做独立应用,进程可看作独立应用
进程有独立地址空间,相互不影响,线程只是进程不同执行路径线程没有独立地址空间,多进程程序比多线程程序更健壮
进程切换开销更大
java进程和线程的关系
java对操作系统的功能进行封装,包括进程和线程
运行一个程序会产生一个进程,进程包括至少一个线程每个进程对应一个JVM实例,多个线程共享JVM里的堆Java采用单线程编程模型,程序会自动创建主线程
主线程可以创建子线程,原则上后于子线程完毕执行
Thread中的start和run的区别
start调用的是start0方法---jvm的JVM_StartThread方法----创建新的thread 之后调用run()方法启动线程
所以调用start()方法会创建一个新的子线程并启动,run()方法只是Thread的一个普通方法的调用,所以run()只是调用了当前线程方法。
二者不存在可比性
Thread和Runnable关系
Thread实现了Runnable接口,使run支持多线程
因为类单一继承原则,推荐多使用Runnable接口
如何给run()方法传参
构造函数/成员变量/回调函数
如何实现处理线程的返回值
主线程等待法
使用Thread类的join()阻塞当前线程以等待子线程处理完毕
![]() |
通过Callable接口实现:通过FutureTask Or线程池获取
![]() |
![]() |
没有显式等待,但会在is.Done没有返回true之前,一直等待,直到等了5秒,获取到了返回的value才会打印
先建立一个线程池,提交任务到线程池,返回Future类型对象,调用isDone()和get()方法线程池可以提交多个任务并发执行
线程的状态
六个状态:
新建NEW:创建后尚未启动的线程的状态
运行Runnable:包含Running(可运行线程中,等待被CPU选中)和Ready(线程池中,被选中就进入Running状态)
无限制等待Waiting:不会被分配CPU执行时间,需要显式被唤醒
(没有设置Timeout参数的Object.wait()方法///没有设置Timeout参数的Thread.join()方法///LockSupport.park()方法)
限期等待Time Waiting:在一定时间后会由系统自动唤醒
(设置Timeout参数的Object.wait()方法///设置Timeout参数的Thread.join()方 法///LockSupport.parkNanos()方法///LockSupport.parkUntil()方法)
阻塞Blocked:等待获取排他锁
结束状态Terminated:已终止线程的状态,线程已经结束执行
sleep和wait区别
sleep是Thread类的方法
Thread.sleep只会让出CPU,不会导致锁行为的改变Object.wait不仅让出CPU,还会释放已经占有的同步资源锁
notify和notifyAll区别
锁池
假如线程A已经拥有某个对象(不是类)的锁,而其它线程B/C想要调用这个对象的某个synchronized方法(或者块),由于B/C线程在进入对象的synchronized方法之前必须先获得该对象锁 的拥有权,而恰巧该对象的锁目前正在被线程A所占用,此时B/C线程就会被阻塞,进入该对象的锁池去等待
等待池
假设线程A调用了某个对象的wait()方法,线程A就会释放该对象的锁,同时线程A就进入到该对象的等待池中,进入到等待池中的线程不回去竞争该对象的锁。
notifyAll会让全部处于等待池的线程全部进入锁池去竞争获取锁。(没得到锁的线程会留在锁池而不是回到等待池
notifyAll会随机挑选一个处于等待池的线程全部进入锁池去竞争获取锁。
yield
当调用Thread.yield()函数,会给线程调度器一个当前线程愿意让出CPU使用的暗示,但是线程调度 器可能会忽视这个暗示
yield不会对锁状态有影响
如何中断线程
stop()已经被抛弃
调用interrupt(),通知线程应该中断了
如果线程处于阻塞状态,那么线程将立即退出被阻塞状态,并抛出一个InterruptedException异常
如果线程处于正常活动状态,那么会将该线程额中断标志设置为true。被设置中断标志的线程将继续正常运行,不受影响
需要被调用线程配合中断
在正常运行任务时,经常检查本线程的中断标志位,如果被设置了中断标志就自行停止线程
如果线程处于正常活动状态,那么会将该线程额中断标志设置为true。被设置中断标志的线程将继续正常运行,不受影响
这里的逻辑是新建线程(状态为new),start线程(Timed waiting),没设置为中断就i++,之后打印i 的值和线程状态。休眠300毫秒中断
这个检查到中断,并且线程位于阻塞状态,则catch异常并中断。但继续打印发现线程还是RUNNABLE状态,所以这就是“被设置中断标志的线程将继续正常运行,不受影响”,过300毫秒继续打印,发现线程终止。
线程状态以及线程状态之间的转换
通过thread或继承Runnable接口new出线程即新建
新建的线程调用start方法进入Runnable状态,可运行状态
Runnable状态被os时间片选中进入Running状态,运行状态
Running状态调用yield方法可能会让出cpu进入Runnable状态,也可能是时间片用完了,也会进 入Runnable状态
Running状态如果等待用户输入或者调用sleep会进入阻塞状态,不会释放锁。
Runnable状态和Running状态如果调用synchronized方法,发现自己没权限,进入锁池等待Running状态调用wait会进入等待队列,如果被notifyAll或者notify唤醒,就进入锁池
锁池获得锁,就进入Runnable状态
Running状态执行完毕或者异常退出,就进入死亡状态TERMINATED