在java的并发编程之中,有几个比较重要的模式,其中notify-wait的模式显得很重要,因为他是经典的生产者-消费者模式实现的基础,而在生产者-消费者模式上,又可以演变出很多的设计方法,比如java的线程池就是典型的生产者消费者模式,今天就来说一下notify-wait
首先说一下wait在notify之后的的执行逻辑
在Java对象中,有两种池
琐池-----------------------synchronized
等待池---------------------wait(),notify(),notifyAll()
如果一个线程调用了某个对象的wait方法,那么该线程进入到该对象的等待池中(并且已经将锁释放),
如果未来的某一时刻,另外一个线程调用了相同对象的notify方法或者notifyAll方法,
那么该等待池中的线程就会被唤起,然后进入到对象的锁池里面去获得该对象的锁, 如果获得锁成功后,那么该线程就会沿着wait方法之后的路径继续执行。注意是沿着wait方法之后
notify-wait需要在同步的环境下使用,一般套用的方法是:
1,获取对象的锁
2,如果条件不满足,那么调用对象的wait方法,被通知后仍然要检查条件,之所以需要条件检查,是因为被通知后仍然需要获取对象锁,也因此,一般条件检查部分都需要使用while循环来检查
3,条件满足则执行下面对应的逻辑
伪码如下:
synchronized(对象){
while(条件不满足){
对象.wait();
}
对应的处理逻辑
}
通知方遵循的规则如下:
1,获取对象的锁
2,改变现有条件
3,通知所有等待在对象上的线程
synchronized(对象){
改变条件
对象.notify();
}
话不多说,上代码分析
package com.luchi.threadPoolSimulation; import java.sql.Connection; import java.util.LinkedList; public class ConnectionPool { private LinkedList<Connection> pool=new LinkedList<>(); public ConnectionPool(int initialSize){ //初始化threadPool for(int i=0;i<initialSize;i++){ pool.addLast(ConnectionDriver.createConnection()); } } public void releaseConnection(Connection conn){ if(conn!=null){ synchronized (pool) { //获取锁 pool.addLast(conn); //更改条件 pool.notifyAll(); //通知 } } } public Connection fetchConnection(int mill) throws InterruptedException{ synchronized (pool) { //获取锁 if(mill<=0){ while(pool.isEmpty()){ pool.wait(); } return pool.getLast(); }else{ long Future=System.currentTimeMillis()+mill; long remain=mill; while(pool.isEmpty() && remain>0){ //循环检查条件 pool.wait(remain); //条件不满足陷入等待 remain=Future-System.currentTimeMillis(); } //条件满足或者超时执行下面逻辑 Connection result=null; if(!pool.isEmpty()){ result=pool.removeFirst(); } return result; } } } }
代码说明:
代码模拟java的数据库连接池,其中fetchConnection是取connection连接,releaseConnection是释放连接,代码的构造函数初始化了这个connection连接池的大小。
首先来看一下取连接:
取连接加入了超时检查模型,也就是说如果在没有取得连接的情况下过了指定的一段时间,将不会再次陷入wait之中而是接着执行(返回null),代码完全遵守之前描述的等待原则,先获取锁,然后循环检查条件,执当检查条件满足或者超时的情况下接着往下面行
然后再看一下释放连接的releaseCOnnection,同样还是遵循之前描述的模型,获取锁-更改条件-执行逻辑
了解这这个,下回就可以分析JDK1.7中的线程池的源码了,over。
相关推荐
总结来说,Java的`wait()`、`notify()`和`notifyAll()`方法是实现多线程间协作的关键工具,特别是对于解决生产者消费者问题。理解它们的工作原理和正确使用方式对于编写高效的并发代码至关重要。在上述示例中,我们...
7. **Java并发工具类**:除了`wait()`和`notify()`,Java并发库提供了更高级的工具,如`BlockingQueue`,它提供了线程安全的队列操作,简化了生产者消费者模式的实现,避免了直接使用`wait()`和`notify()`可能导致的...
`wait`和`notify`通常与条件变量模式结合使用,以实现线程之间的协调。例如,在生产者-消费者模型中,生产者线程在向共享缓冲区添加元素时可能会调用`wait()`,直到有空闲空间可用。消费者线程在移除元素后会调用`...
生产者和消费者模式可以使用wait和notify方法来实现。 实现生产者和消费者模式时,需要使用synchronized关键字来确保线程安全。在生产者线程中,使用wait方法来等待仓库容量达到一定数量后再继续生产。在消费者线程...
在探索多线程和设计模式的同时,还要关注线程间的通信(如使用`wait()`、`notify()`、`notifyAll()`等)、同步机制(如`synchronized`关键字、`Lock`接口等)、线程池的使用(`ExecutorService`、`...
从单词意思来看,wait就是等待,说明这个资源没有准备好,我要等,还有这一个wait(long timeout) ,表示我只能等待你这么长时间了,过时不候啊,而调用notify()的线程肯定就是对资源进行处理的,处理完进行通知。...
在Java中,线程的wait()、notify()和notifyAll()方法体现了状态模式。 15. **职责链模式**:避免请求的发送者和接收者之间的耦合,将多个可能的处理者串成一条链,请求沿着链传递,直到有处理者处理为止。 这些...
Java中的wait/notify/notifyAll可用来实现线程间通信,是Object类的方法,这三个方法都是native方法,是平台相关的,常用来实现生产者/消费者模式。 wait()方法将当前线程置于等待状态,直到其他线程调用notify()或...
- 多线程编程:通过Thread类或Runnable接口创建线程,理解同步机制如synchronized关键字、wait/notify机制。 - 异常处理:学习如何使用try-catch-finally语句块,以及自定义异常类。 - IO流操作:掌握字节流、...
例如,在生产者消费者模式中,生产者线程可以通过notify()方法唤醒消费者线程,而消费者线程可以通过wait()方法等待生产者线程的唤醒。 在上面的代码中,我们可以看到一个经典的生产者消费者模式的实现。生产者线程...
wait set——线程的休息室 wait方法——把线程放入wait set notify方法——从wait set拿出线程 notifyAll方法——从wait set拿出所有线程 wait、notify、notifyAll是Object类的方法 线程的状态移转 跟线程有关的其他...
- 必须在 synchronized 代码块或方法中使用 wait()、notify() 和 notifyAll(),否则会抛出 `IllegalMonitorStateException`。 3. **sleep() 方法**: - Thread.sleep() 方法让当前线程暂停执行指定的时间,然后...
【生产者消费者模式】是一种经典的多线程设计模式,用于解决资源的共享和并发访问问题。...在Java中,我们可以选择wait() / notify()、await() / signal()或BlockingQueue来实现这一模式,根据具体场景选择合适的方法。
Java中的`Object.wait()`、`Object.notify()`以及`Object.notifyAll()`是多线程编程中的核心方法,它们用于实现线程间的协作与通信。这些方法是`Object`类的最终原生(`final native`)方法,这意味着它们的实现位于...
- wait()、notify()、notifyAll()的理解与使用 - Java内存模型(JMM)和线程通信 - Callable与Future接口在多线程中的应用 4. **网络编程**: - TCP与UDP的区别 - Socket编程的基本原理 - HTTP协议的工作流程...
2. **线程的唤醒**:`sleep()`方法的线程在指定的时间结束后会自动恢复执行,而`wait()`方法的线程需要被其他线程显式地唤醒(`notify()`或`notifyAll()`方法)。 3. **调用位置**:`sleep()`可以在任何地方调用,而`...
1.notify的使用 2. wait的使用 3. linux系统下的开发 4.多线程通信 5. 生产者和消费者模式
- **线程状态变化**:当调用 `wait()` 时,当前线程会从运行状态 (`RUNNABLE`) 转变为等待状态,并加入到该对象的等待队列中。 - **释放锁**:调用 `wait()` 的一个重要特点是会释放当前持有的锁,允许其他线程进入...
- `wait()`、`notify()`和`notifyAll()`方法用于线程之间的通信。 - `wait()`使当前线程等待,直到被另一个线程唤醒或等待时间到达。 - `notify()`随机唤醒一个等待线程。 - `notifyAll()`唤醒所有等待线程。 9...