最近看线程,今天复习时发觉自己并没有搞明白sleep,interrupt和wait,notify的区别,以前曾见人这么分析二者:
sleep()方法是使线程停止一段时间的方法。在sleep 时间间隔期满后,线程不一定立即恢复执行。这是因为在那个时刻,其它线程可能正在运行而且没有被调度为放弃执行,除非(a)"醒来"的线程具有更高的优先级。 (b)正在运行的线程因为其它原因而阻塞。
wait()是线程交互时,如果线程对一个同步对象x 发出一个wait()调用,该线程会暂停执行,被调对象进入等待状态,直到被唤醒或等待时间到。
但我还是不怎么明白,就跑去论坛上找文章,看到了下面这篇,讲的不错,收录下来。
下面是我原来在CJSDN论坛上看到的一个贴子,涉及到同步,wait(),notify()等概念的理解,我试着根据原来的一些回复和Think in Java上的相关概念将wait()和notify()这两个方法剖析了一下,欢迎指教.
问题如下:
file://分析这段程序,并解释一下,着重讲讲synchronized、wait(),notify 谢谢!
class ThreadA
{
public static void main(String[] args)
{
ThreadB b=new ThreadB();
b.start();
System.out.println("b is start....");
synchronized(b)//括号里的b是什么意思,起什么作用?
{
try
{
System.out.println("Waiting for b to complete...");
b.wait();//这一句是什么意思,究竟让谁wait?
System.out.println("Completed.Now back to main thread");
}catch (InterruptedException e){}
}
System.out.println("Total is :"+b.total);
}
}
class ThreadB extends Thread
{
int total;
public void run()
{
synchronized(this)
{
System.out.println("ThreadB is running..");
for (int i=0;i<100;i++ )
{
total +=i;
System.out.println("total is "+total);
}
notify();
}
}
}
要 分析这个程序,首先要理解notify()和wait(),为什么在前几天纪录线程的时候没有纪录这两个方法呢,因为这两个方法本来就不属于Thread 类,而是属于最底层的object基础类的,也就是说不光是Thread,每个对象都有notify和wait的功能,为什么?因为他们是用来操纵锁的, 而每个对象都有锁,锁是每个对象的基础,既然锁是基础的,那么操纵锁的方法当然也是最基础了.
再往下看之前呢,首先最好复习一下Think in Java的14.3.1中第3部分内容:等待和通知,也就是wait()和notify了.
按照Think in Java中的解释:"wait()允许我们将线程置入"睡眠"状态,同时又"积极"地等待条件发生改变.而且只有在一个notify()或notifyAll()发生变化的时候,线程才会被唤醒,并检查条件是否有变."
我们来解释一下这句话.
"wait()允许我们将线程置入"睡眠"状态",也就是说,wait也是让当前线程阻塞的,这一点和sleep或者suspend是相同的.那和sleep,suspend有什么区别呢?
区 别在于"(wait)同时又"积极"地等待条件发生改变",这一点很关键,sleep和suspend无法做到.因为我们有时候需要通过同步 (synchronized)的帮助来防止线程之间的冲突,而一旦使用同步,就要锁定对象,也就是获取对象锁,其它要使用该对象锁的线程都只能排队等着, 等到同步方法或者同步块里的程序全部运行完才有机会.在同步方法和同步块中,无论sleep()还是suspend()都不可能自己被调用的时候解除锁 定,他们都霸占着正在使用的对象锁不放.
而wait却可以,它可以让同步方法或者同步块暂时放弃对象锁,而将它暂时让给其它需要对象锁的人(这里应该是程序块,或线程)用,这意味着可在执行wait()期间调用线程对象中的其他同步方法!在其它情况下(sleep啊,suspend啊),这是不可能的.
但是注意我前面说的,只是暂时放弃对象锁,暂时给其它线程使用,我wait所在的线程还是要把这个对象锁收回来的呀.wait什么?就是wait别人用完了还给我啊!
好,那怎么把对象锁收回来呢?
第一种方法,限定借出去的时间.在wait()中设置参数,比如wait(1000),以毫秒为单位,就表明我只借出去1秒中,一秒钟之后,我自动收回.
第二种方法,让借出去的人通知我,他用完了,要还给我了.这时,我马上就收回来.哎,假如我设了1小时之后收回,别人只用了半小时就完了,那怎么办呢?*!当然用完了就收回了,还管我设的是多长时间啊.
那么别人怎么通知我呢?相信大家都可以想到了,notify(),这就是最后一句话"而且只有在一个notify()或notifyAll()发生变化的时候,线程才会被唤醒"的意思了.
因此,我们可将一个wait()和notify()置入任何同步方法或同步块内部,无论在那个类里是否准备进行涉及线程的处理。而且实际上,我们也只能在同步方法或者同步块里面调用wait()和notify().
这个时候我们来解释上面的程序,简直是易如反掌了.
synchronized (b){...};的意思是定义一个同步块,使用b作为资源锁。b.wait();的意思是临时释放锁,并阻塞当前线程,好让其他使用同一把锁的线程有机 会执行,在这里要用同一把锁的就是b线程本身.这个线程在执行到一定地方后用notify()通知wait的线程,锁已经用完,待notify()所在的 同步块运行完之后,wait所在的线程就可以继续执行.
分享到:
相关推荐
这两个方法是Java中实现线程间通信的重要手段之一,尤其在解决生产者消费者模型、读者写者问题等经典同步问题时非常有用。 #### 代码分析 给出的代码示例包括两个类:`ThreadA`和`ThreadB`。`ThreadA`作为主程序...
在实际开发中,`wait()`, `notify()`和`notifyAll()`常用于实现生产者-消费者模型、哲学家就餐问题等经典并发问题。它们提供了一种精细控制线程间交互的方式,但使用时需谨慎,避免死锁和活锁的发生。 总之,理解和...
- **生产者-消费者模式**:在经典的生产者-消费者模型中,生产者生产数据,消费者消费数据。当缓冲区满时,生产者必须等待消费者消费掉一部分数据才能继续生产;同样,当缓冲区为空时,消费者必须等待生产者生产数据...
生产者-消费者问题是多线程编程中的一个经典问题。该问题包含两个线程:生产者线程负责生成数据并放入队列或缓冲区中,消费者线程负责从队列或缓冲区中取出数据进行处理。为了保证数据的完整性和一致性,必须确保...
在实际应用中,这样的接口可能会被用在例如生产者消费者模型、读者写者问题等经典的并发问题解决方案中。通过调用`notify_next_wait3_func`,生产者可以在完成生产后通知消费者可以开始消费,或者写者在完成写操作后...
oracle数据库的经典调优文档,建议dba深入阅读
生产者-消费者问题是多线程编程中的经典问题,其中生产者线程生成数据,而消费者线程消耗这些数据。为确保生产者不会在没有消费者准备接收时生产过多数据,以及消费者不会尝试消费未生成的数据,需要一种机制来同步...
NEH算法,全称是Nemhauser-Ellis-Holzer算法,是一种经典的启发式调度算法,主要用于解决多工件、多机器的调度问题。此算法由Nemhauser、Ellis和Holzer在1975年提出,主要通过贪心策略来构建初始解,并逐步优化以...
在上面的代码中,我们可以看到一个经典的生产者消费者模式的实现。生产者线程通过notify()方法唤醒消费者线程,而消费者线程通过wait()方法等待生产者线程的唤醒。这种方式可以实现线程之间的通信和同步,从而实现...
本资源摘要信息涵盖了操作系统中经典的进程同步问题,包括使用信号量机制、wait 和 signal 原语描述的多种同步问题。 1. 测量控制系统中的数据采集任务和计算任务之间的同步问题 在这个问题中,我们使用信号量机制...
6. **哲学家就餐问题**:这是一个经典的多线程问题,可以使用生产者消费者模型来解决。每个哲学家既是生产者(思考问题),也是消费者(吃饭),通过共享筷子(资源)来实现同步。 7. **Java并发工具类**:除了`...
4. **多线程**:Java内置了对多线程的支持,经典代码可能涉及Thread类的使用,线程同步(synchronized关键字,wait/notify机制),以及ExecutorService和Future接口的高级用法。 5. **IO流**:Java的IO流系统强大而...
"WAIT - an Information Retrieval Toolbox" 是一个开源的信息检索工具箱,主要基于Perl和XS语言进行重写了经典的freeWAIS-sf搜索引擎。这个项目旨在提供一个高效、灵活且可定制的解决方案,用于处理大规模文本数据...
操作系统实验报告--经典的生产者—消费者问题 本实验报告的主要内容是模拟经典的生产者—消费者问题,以便更好地理解进程同步问题。生产者—消费者问题是典型的 PV 操作问题,假设系统中有一个比较大的缓冲池,生产...
1.3 notify、notifyAll的区别2 两个比较经典的使用案例2.1 案例1 — ABCABC。。。三个线程顺序打印问题2.1.1 题目2.1.2 题目分析2.1.3 我的答案2.2 生产者消费者问题2.2.1 题目2.2.2 题目分析2.2.3 我的答案 源码...
在多线程环境下,理解和运用synchronized关键字、wait()、notify()方法以及并发工具类(如Semaphore、CountDownLatch)是必不可少的。UserList在处理大量用户数据时,可能需要用到这些技术来确保数据的一致性和程序...
此外,异常处理、多线程编程、并发控制(如synchronized关键字、wait/notify机制)、线程池等高级主题也会在书中有所涉及。这些内容对于编写健壮、高效的应用程序非常重要。 最后,书中还会涵盖一些实用的Java库,...
在Linux环境下,我们可以使用POSIX信号量,它们提供了`sem_init()`、`sem_post()`(P操作,增加信号量值)和`sem_wait()`(V操作,减少信号量值)等函数来实现这些同步机制。互斥信号量用于实现互斥访问,而条件变量...
6. **多线程**:讲解线程的创建方式(Thread类和Runnable接口)、线程同步(synchronized关键字、wait/notify机制、Lock锁)以及并发工具类如Semaphore、CountDownLatch等。 7. **网络编程**:包括Socket编程,理解...