图解
6种状态
new , runnable , blocked , waiting , timed waiting , terminated
blocked,waiting,timed waiting 我们都称为阻塞状态
- 当进入synchronized同步代码块或同步方法中,且没有获取到锁,线程就进入blocked状态,知道锁被释放,重新进入runnable状态
- 当线程调用wait()或者join时,线程都会进入到waiting状态,当调用notify或notifyAll时,或者join的线程执行结束后,会进入runnable状态
- 当线程调用sleep(time),或者wait(time)时,进入timed waiting状态,当休眠时间结束后,或者调用notify或notifyAll时会重新runnable状态。
- 程序执行结束,线程进入terminated状态
线程安全
当多个线程访问一个对象时,如果不进行额外的同步控制或其他协调的操操作,调用这个对象的行为都可以获取正确的结果,我们就说这个对象是线程安全的
Sleep 和 wait 的区别
- 所属的类不同 :
sleep方法是定义在Thread上
wait方法是定义在Object上- 对于锁资源的处理方式不同
sleep不会释放锁
wait会释放锁- 使用范围:
sleep可以使用在任何代码块
wait必须在同步方法或同步代码块执行
为什么wait必须写在同步代码块中?
原因是避免CPU切换到其他线程,而其他线程又提前执行了notify方法,那这样就达不到我们的预期(先wait再由其他线程来唤醒),所以需要一个同步锁来保护
为什么wait要定义在Object中,而不定义在Thread中?
在同步代码块中,我们说需要一个对象锁来实现多线程的互斥效果,也就是说,Java的锁是对象级别的,而不是线程级别的。