Java之Thread.sleep()浅析

在java开发中,经常会用到Thread.sleep(1000);类似的语句来使线程挂起一段时间,那么底层的逻辑是怎么操作的呢?
问题1:假设现在时间是2022-1-1 8:00:00.000,若是调用了Thread.sleep(1000); 在2022-1-1 8:00:01.000 时,这个线程会不会继续执行?
问题2:Thread.sleep(1000);是挂起了0毫秒,那与去掉这句代码有什么区别?
说起线程,其底层就要了解一下操作系统原理了,操作系统主要使用window和linux,windows中CPU竞争的策略是抢占式的,linux中CPU竞争策略是轮询的。
轮询策略中,所有的线程排成一个队列,操作系统按照它们的顺序,给每个线程分配一段时间,若是时间结束时线程还在运行,则CPU将被剥夺分配给另一个线程;若是在时间片结束前阻塞或结束,则CPU当即切换。当线程用完它的时间片后,被移动到队列的末尾。
抢占式策略中,所有的线程按照优先级排成一个队列,操作系统会把CPU交给优先级最高的线程,当一个线程获得了CPU时间,除非自己放弃使用CPU,否则将完全占用。当线程执行结束或自己挂起后,操作系统会重新计算一次优先级进行排列,然后把CPU交给队列中优先级最高的。

场景说明两种策略,吃蛋糕,一个刀叉(一个CPU),10个小伙伴(10个线程)

linux的轮询策略,每个人上来吃1分钟,时间到了换下一个,最后一个吃完了再从头开始。该种策略不管饥饿程度(优先级)的情况、饭量情况。若有人不饿、饭量小,吃了30秒就吃饱了,可以根操作系统说,吃饱了(Thread.sleep()挂起),于是换下一个。
windows抢占式策略,根据饥饿程度(优先级)的情况、饭量情况进行排列,最前面的人先上来吃,吃到不想吃为止,等这个人吃完了,再重新排列,再让最前面的人上来吃。若有人吃了30秒说不想吃了(Thread.sleep()挂起),操作系统重新计算进行排列。若有人吃了30秒说1小时之内都不要叫我上来吃了,就告诉操作系统,在未来多少毫秒内不参与CPU竞争。

问题1,不一定。因为你告诉操作系统,在未来1000毫秒内不参与CPU竞争,那么1000毫秒过去之后:轮询策略下,这时候也行另一个线程正在使用CPU,那么这时操作系统是不会重新分配CPU的,直到那个线程挂起或结束;抢占式策略下,这时候操作系统正进行CPU分配,那么当前线程也不一定是优先级最高的那个,CPU还是可能被其他优先级高的线程抢占去。
问题2,有。吃蛋糕的场景里,有个女同学的优先级最高(最漂亮),操作系统总是叫她上来吃,而且女同学也非常喜欢吃蛋糕,但又很善良,没吃几口就想,会不会有很饿的同学,那么我就让给他。因此没吃几口就跟操作系统说重新计算一下优先级,在未来0毫秒内不要叫我上来吃了。注意,此时是连同女同学本人也参与计算的,因为0毫秒已经过去了,所有若没有很饿的同学,那么下一次还是她被叫上来吃蛋糕。因此,Thread.sleep(0)的作用,就是“触发操作系统立刻重新进行一次CPU竞争”。竞争的结果也许是当前线程仍然获得CPU控制权,也许会换成别的线程获得CPU控制权。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 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. 使用Object的wait()和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、付费专栏及课程。

余额充值