文章目录
Java_多线程那点事
Java内多线程函数含义
synchronized
- Java中每一个对象都可以成为一个监视器
(Monitor)
, 该Monitor由一个锁(lock)
, 一个等待队列(waiting queue)
, 一个入口队列(entry queue)
组成。 - 对于一个对象的方法, 如果没有synchronized关键字修饰, 该方法可以被任意数量的线程,在任意时刻调用。
对于添加了synchronized关键字的方法,任意时刻只能被唯一的一个获得了对象实例锁的线程调用。 - synchronized用于实现多线程的同步操作
Wait
wait()
,notify()
,notifyAll()
和synchronized
需要搭配使用, 用于线程同步wait()
总是在一个循环中被调用,挂起当前线程来等待一个条件的成立。 Wait调用会一直等到其他线程调用notifyAll()
时才返回。wait()
在其他条件不满足时,将获得的锁释放掉- 当一个线程在执行
synchronized
的方法内部,调用了wait()
后, 该线程会释放该对象的锁, 然后该线程会被添加到该对象的等待队列中(waiting queue)
, 只要该线程在等待队列中, 就会一直处于阻塞状态,不会被调度执行。 要注意wait()
方法会强迫线程先进行释放锁操作,所以在调用wait()
时, 该线程必须已经获得锁,否则会抛出异常。但是由于wait()
在synchronized
的方法内部被执行, 锁一定已经获得, 就不会抛出异常了。
Notify
- 当一个线程调用一个对象的
notify()
方法时, 调度器会从所有处于该对象等待队列(waiting queue)
的线程中取出任意一个线程, 将其添加到入口队列(entry queue)
中. 然后在入口队列中的多个线程就会竞争对象的锁, 得到锁的线程就可以继续执行。 如果等待队列中(waiting queue)
没有线程,notify()
方法不会产生任何作用 notifyAll()
和notify()
工作机制一样, 区别在于notifyAll()
会将等待队列(waiting queue)
中所有的线程都添加到入口队列中(entry queue)
notifyAll()
比notify()
更加常用, 因为notify()
方法只会唤起一个线程,且无法指定唤醒哪一个线程
多线程的几种状态
-
新建
- 当一个线程对象被创建,但还未调用 start 方法时处于新建状态
- 此时还只是Java对象,还未与操作系统底层线程关联
-
可运行
- 调用了 start 方法,就会由新建进入可运行
- 此时与底层线程关联,由操作系统调度执行
-
终结
- 线程内代码已经执行完毕,由可运行进入终结
- 此时会取消与底层线程关联
-
阻塞
- 当获取锁失败后,由可运行进入 Monitor 的阻塞队列阻塞,此时不占用 cpu 时间
- 当持锁线程释放锁时,会按照一定规则唤醒阻塞队列中的阻塞线程,唤醒后的线程进入可运行状态
-
等待