Skip to main content

线程的状态和锁

threadthread线程状态About 3 min

简单记录一下多线程的东西

线程的状态

内容来自JAVA 线程状态及转化open in new window

线程共包括以下5种状态:

  1. 新建状态(New): 线程对象被创建后,就进入了新建状态。例如,Thread thread = new Thread()。
  2. 就绪状态(Runnable): 也被称为“可执行状态”。线程对象被创建后,其它线程调用了该对象的start()方法,从而来启动该线程。例如,thread.start()。处于就绪状态的线程,随时可能被CPU调度执行。
  3. 运行状态(Running): 线程获取CPU权限进行执行。需要注意的是,线程只能从就绪状态进入到运行状态。
  4. 阻塞状态(Blocked): 阻塞状态是线程因为某种原因放弃CPU使用权,暂时停止运行。直到线程进入就绪状态,才有机会转到运行状态。
    阻塞的情况分三种:
    (01) 等待阻塞 -- 通过调用线程的wait()方法,让线程等待某工作的完成。
    (02) 同步阻塞 -- 线程在获取synchronized同步锁失败(因为锁被其它线程所占用),它会进入同步阻塞状态。
    (03) 其他阻塞 -- 通过调用线程的sleep()或join()或发出了I/O请求时,线程会进入到阻塞状态。当sleep()状态超时、join()等待线程终止或者超时、或者I/O处理完毕时,线程重新转入就绪状态。
  5. 死亡状态(Dead): 线程执行完了或者因异常退出了run()方法,该线程结束生命周期。

2020-04-06-19-04-29

yield() 与 wait()的比较

  1. 状态切换不同: wait()是让线程由“运行状态”进入到“等待(阻塞)状态”,而不yield()是让线程由“运行状态”进入到“就绪状态”。
  2. 是否释放锁: wait()是会线程释放它所持有对象的同步锁,而yield()方法不会释放锁。

为什么notify(), wait()等函数定义在Object中,而不是Thread中

notify(), wait()依赖于“同步锁”,而“同步锁”是对象锁持有,并且每个对象有且仅有一个!这就是为什么notify(), wait()等函数定义在Object类,而不是Thread类中的原因。

乐观锁和悲观锁

悲观锁: 处理之前获取锁并进行加锁, 处理之后进行释放. 持有锁的过程中, 数据不可以被其他线程修改.
乐观锁: 只有在提交时, 进行数据的判断(通过版本号, 主要是数据库的概念), 如果提交失败可选择重试, 抛出异常等操作.

公平锁和非公平锁

公平锁: 按照请求锁的时间先后进行线程处理.
非公平锁: 没有时间顺序.

独占锁和共享锁

锁能否被多个线程共同持有. 独占锁是一种悲观锁, 共享锁是一种乐观锁.

可重入锁

线程是否可以再次获取已经获取的锁.
ReentrantLock, synchronized都是可重入锁

自旋锁

当一个线程在获取锁的时候,如果锁已经被其它线程获取,该线程不会立刻阻塞, 而是不断尝试获取锁,直到获取到锁才会退出循环。
-XX:PreBlockSpinsh可以设置获取的次数, 默认值为10. 只有在没有获取到的情况下才会进入阻塞状态.

自旋锁是使用cpu时间换区线程阻塞与调度的开销, 但是这些cpu时间很有可能被浪费掉.

What do you think?
  • 0
  • 0
  • 0
  • 0
  • 0
  • 0
Comments
  • Latest
  • Oldest
  • Hottest
Powered by Waline v3.0.0-alpha.10