java之Thread.sleep(long)与object.wait()/object.wait(long)的区别及相关概念梳理(good)

 

一、Thread.sleep(long)与object.wait()/object.wait(long)的区别
sleep(long)与wait()/wait(long)行为上有些类似,主要区别如下:
1.Thread.sleep(long)是属于Thread类的静态方法。其基本语义是使当前运行的线程暂停一段时间。实现细节是把当前线程放入就绪线程队列中,直到睡眠时间到期才可被调度为执行线程(在时间到期前无法被调度为执行线程)。此方法可以在sychronized代码块中,调用此方法不释放对象锁;也可以不在sychronized代码块中。此方法的作用线程一定是当前运行的线程(如果代码在一个线程类中,不一定是代码所在的线程实例),即使是在线程对象上调用sleep(long)或调用Thread.currentThread().sleep(long)。


2.object.wait()是属于类实例(Object及其子类实列,也包括Class类实例)的方法。
其基本语义是使当前线程等待,直到被通知,默认是this.wait()。实现细节是把当前线程放入阻塞线程队列中,并把当前线程注册为指定对象的监听器,并锁释放指定对象的锁;当被notify/notifyAll通知时,重新争取指定对象的锁,并把当前线程从指定对象的监听器中移除,把当前线程从阻塞队列放入就绪队列,等待被调度。
此方法必须在sychronized代码块中,且锁定的对象要与等待通知来源的对象一致。而wait(long)方法阻塞时放入的是就绪队列,等待时间到期或被通知就可被调度,其他与wait()方法相同。
如果当前线程不是对象所得持有者,该方法抛出一个java.lang.IllegalMonitorStateException 异常”
http://blog.csdn.net/intlgj/article/details/6245226

注:从上可基本认识到线程的执行、就绪、阻塞三种状态的切换,以及线程的调度(操作系统调度线程)和就绪线程队列、阻塞线程队列(实际实现可能更复杂,比如优先级,调度策略等)。可认识到object(包括Class类的实例)的wait/notify/notifyAll的对象监听和通知事件模式,以及对象上的锁、锁队列(实际实现可能更复杂,比如优先级,锁争用等)、阻塞线程监听队列(notify时只通知一个监听器,具体调度未知,另有自动唤醒,条件唤醒)。

http://blog.csdn.net/qingmingcom/article/details/6590967

 

    wait()方法与notify()必须要与synchronized(resource)一起使用。也就是wait与notify针对已经获取了resource锁的线程进行操作
从语法角度来说就是Obj.wait(),Obj.notify必须在synchronized(Obj){...}语句块内。
从功能上来说wait()线程在获取对象锁后,主动释放CPU控制权,主动释放对象锁,同时本线程休眠。直到有其它线程调用对象的notify()唤醒该线程,才能继续获取对象锁,并继续执行。
相应的notify()就是对对象锁的释放操作
【因此,我们可以发现,wait和notify方法均可释放对象的锁,但wait同时释放CPU控制权,即它后面的代码停止执行,线程进入阻塞状态,而notify方法不立刻释放CPU控制权,而是在相应的synchronized(){}语句块执行结束,再自动释放锁。】
释放锁后,JVM会在等待resoure的线程中选取一线程,赋予其对象锁,唤醒线程,继续执行。这样就提供了在线程间同步、唤醒的操作。
Thread.sleep()与Object.wait()二者都可以暂停当前线程,释放CPU控制权,主要的区别在于Object.wait()在释放CPU同时,释放了对象锁的控制
而在同步块中的Thread.sleep()方法并不释放锁,仅释放CPU控制权。

  1)wait()、notify()和notifyAll()方法是本地方法,并且为final方法,无法被重写。

  2)调用某个对象的wait()方法能让当前线程阻塞,并且当前线程必须拥有此对象的monitor(即锁,或者叫管程)

  3)调用某个对象的notify()方法能够唤醒一个正在等待这个对象的monitor的线程,如果有多个线程都在等待这个对象的monitor,则只能唤醒其中一个线程;

  4)调用notifyAll()方法能够唤醒所有正在等待这个对象的monitor的线程;

     在Java中,是没有类似于PV操作、进程互斥等相关的方法的。JAVA的进程同步是通过synchronized()来实现的,需要说明的是,Java的synchronized()方法类似于操作系统概念中的互斥内存块,在Java中的Object类对象中,都是带有一个内存锁的,在有线程获取该内存锁后,其它线程无法访问该内存,从而实现Java中简单的同步、互斥操作。明白这个原理,就能理解为什么synchronized(this)与synchronized(static XXX)的区别了,synchronized就是针对内存区块申请内存锁,this关键字代表类的一个对象,所以其内存锁是针对相同对象的互斥操作,而static成员属于类专有,其内存空间为该类所有成员共有,这就导致synchronized()对static成员加锁,相当于对类加锁,也就是在该类的所有成员间实现互斥,在同一时间只有一个线程可访问该类的实例。如果需要在线程间相互唤醒就需要借助Object类的wait()方法及nofity()方法。

http://blog.csdn.net/lingzhm/article/details/44940823

什么是monitor?

在HotSpot虚拟机中,monitor采用ObjectMonitor实现。

每个线程都有两个ObjectMonitor对象列表,分别为free和used列表,如果当前free列表为空,线程将向全局global list请求分配ObjectMonitor。

ObjectMonitor对象中有两个队列:
_WaitSet 和 _EntryList,用来保存ObjectWaiter对象列表;
_owner指向获得ObjectMonitor对象的线程。

_WaitSet :处于wait状态的线程,会被加入到wait set;
_EntryList:处于等待锁block状态的线程,会被加入到entry set;

ObjectWaiter

ObjectWaiter对象是双向链表结构,保存了_thread(当前线程)以及当前的状态TState等数据, 每个等待锁的线程都会被封装成ObjectWaiter对象。

http://www.jianshu.com/p/f4454164c017

 

我刚开始深入研究多线程,一直认为Object.wait()/Condition.await()让当前线程阻塞的同时,也会释放当前线程对该condition对象的锁。在之前的一些测试代码中也显示wait后,线程上的锁被释放了。
查看Object.wait()API 描述如下:

Causes the current thread to wait until another thread invokes the notify() method or the notifyAll() method for this object. In other words, this method behaves exactly as if it simply performs the call wait(0). 
    The current thread must own this object's monitor. The thread releases ownership of this monitor and waits until another thread notifies threads waiting on this object's monitor to wake up either through a call to the notify method or the notifyAll method.
 The thread then waits until it can re-obtain ownership of the monitor and resumes execution.

其中“the thread releases ownership of this monitor” 说道当前线程会释放这个对象监控器的所有权。

问题一:这里的monitor怎么理解?监视器(monitor)和锁是什么关系? 这个monitor就是一个术语,或者是一个特殊的类(只包含私有域),但是在java中并没有严格遵守这个概念。个人认为可以简单的理解为锁。

同样的,Condition.await()方法中也有类似描述。
The lock associated with this Condition is atomically released and the current thread becomes disabled for thread scheduling purposes...
也说道,调用await()有,这个条件对象关联的锁被“原子级地”释放。。。

问题二:这个原子级的释放是什么意思? “原子级”其实就是为了保证一个操作的完整性,原子级的释放保证了一个原子操作不会因为线程的突然挂起或者说阻塞而破坏这次操作。

这都能说明调用wait()或者await()后,当前线程会释放该条件对象关联的锁吧?


两个线程用object1做 wait/notify, 是这样:
thread1 得到object1 的 monitor, 调用 object1.wait()
- 释放object1 的 monitor, thread1 wait;

thread2 得到 object1 的 monitor, 调用 object1.notify()
- 激活thread1, 释放object1 的 monitor;

thread1 得到 object1 的 monitor, 从object1.wait()返回, thread1接着执行.
关于monitor, 这个是多进程/线程同步的 一个术语, 见:
Operating Systems Design and Implementation, Third Edition
section 2.2

A monitor is a collection of procedures, variables, and data structures that are all grouped together in a special kind of module or package. 
Processes may call the procedures in a monitor whenever they want to, but they cannot directly access the monitor's internal data structures from procedures declared outside the monitor. 
This rule, which is common in modern object-oriented languages such as Java, was relatively unusual for its time, although objects can be traced back to Simula 67.

In all cases, before this method can return the current thread must
re-acquire the lock associated with this condition. When the
thread returns it is guaranteed to hold this lock.

会释放,其他线程执行Condition.signal(),之前的线程会重新获得锁,继续执行,
AbstractQueuedSynchronizer.java 第2040行,释放锁

https://segmentfault.com/q/1010000002390706

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在 Java 中,除了使用 Thread.sleep() 方法来使线程暂停一段时间外,还可以使用以下方法进行替代: 1. Object.wait(long timeout):使当前线程等待一段时间,或者直到其他线程调用该对象的 notify() 或 notifyAll() 方法。这个方法需要在 synchronized 块中使用,并且线程会释放锁。 2. TimeUnit.SECONDS.sleep(long seconds):使用 TimeUnit 类的 sleep() 方法,可以更方便地指定休眠的时间单位。 3. ScheduledExecutorService.schedule(Runnable command, long delay, TimeUnit unit):使用 ScheduledExecutorService 接口的 schedule() 方法,可以在一定时间后执行任务。 注意:使用以上方法替代 Thread.sleep() 时,需要做好异常处理。 ### 回答2: 在Java中,Thread.sleep()方法可以暂停当前线程的执行一段时间。然而,有时候我们需要在不使用Thread.sleep()的情况下实现类似的效果,可以考虑使用以下方法替代: 1. 使用java.util.concurrent包下的DelayQueue类。DelayQueue是一个无界阻塞队列,可以实现延迟执行任务的效果。我们可以创建一个DelayQueue对象,然后将需要延迟执行的任务封装成实现Delayed接口的对象,设置好延迟时间,然后将任务添加到DelayQueue中。后续线程可以从DelayQueue中获取任务执行。 2. 使用java.util.concurrent包下的ScheduledExecutorService接口。ScheduledExecutorService是一个可定时调度执行任务的服务。我们可以使用ScheduledExecutorService的schedule()方法或者scheduleAtFixedRate()方法来替代Thread.sleep()。这些方法中的参数可以指定任务的执行时间或者执行周期。 3. 使用java.lang.Object类的wait()方法和notify()/notifyAll()方法。这是Java中用于线程之间通信的机制。我们可以在一个线程里使用wait()方法让其进入等待状态,然后在其他线程中使用notify()/notifyAll()方法唤醒该线程。通过这种方式,我们可以实现线程的阻塞和唤醒操作,达到替代Thread.sleep()的效果。 综上所述,我们可以根据具体需求选择适合的替代方法来实现替代Thread.sleep()的功能。 ### 回答3: 在Java中,除了使用Thread.sleep()方法外,还可以使用其他方式来实现类似的效果。 1. 使用ScheduledExecutorService定时器: 可以使用ScheduledExecutorService定时器来替代Thread.sleep(),这样更灵活和可控。通过创建一个定时任务,可以指定任务的延迟时间,然后执行需要暂停的任务。示例代码如下: ```java import java.util.concurrent.Executors; import java.util.concurrent.ScheduledExecutorService; import java.util.concurrent.TimeUnit; public class SleepAlternative { public static void main(String[] args) { ScheduledExecutorService executorService = Executors.newSingleThreadScheduledExecutor(); executorService.schedule(() -> { // 执行需要暂停的任务 try { Thread.sleep(1000); // 暂停1秒 System.out.println("任务执行完毕"); } catch (InterruptedException e) { e.printStackTrace(); } }, 1, TimeUnit.SECONDS); executorService.shutdown(); } } ``` 2. 使用Objectwait()和notify()方法: 通过使用Object类的wait()和notify()方法,可以实现线程的等待和唤醒,从而达到类似Thread.sleep()的效果。示例代码如下: ```java public class SleepAlternative { public static void main(String[] args) { Object lock = new Object(); Thread thread = new Thread(() -> { synchronized (lock) { try { lock.wait(1000); // 暂停1秒 System.out.println("任务执行完毕"); } catch (InterruptedException e) { e.printStackTrace(); } } }); thread.start(); } } ``` 通过上述两种方式,可以实现线程的等待和暂停,从而达到替代Thread.sleep()的效果。这些替代方法相对于Thread.sleep()更加灵活和可控,同时可以避免一些潜在的问题,如线程被中断等。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值