C# sleep 和wait的区别

本文详细对比了sleep与wait两种线程暂停方法的区别。sleep属于Thread类,可在任意位置使用,不释放锁;而wait用于释放对象锁并等待通知,需在同步环境中调用。此外,wait需其他线程唤醒。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

sleep和wait都是使线程暂时停止执行的方法,但它们有很大的不同。

1. sleep是线程类Thread 的方法,它是使当前线程暂时睡眠,可以放在任何位置。

而wait,它是使当前线程暂时放弃对象的使用权进行等待,必须放在同步方法或同步块里。
2.Sleep使用的时候,线程并不会放弃对象的使用权,即不会释放对象锁,所以在同步方法或同步块中使用sleep,一个线程访问时,其他的线程也是无法访问的。

而wait是会释放对象锁的,就是当前线程放弃对象的使用权,让其他的线程可以访问。

3.线程执行wait方法时,需要其他线程调用Monitor.Pulse()或者Monitor.PulseAll()进行唤醒或者说是通知等待的队列。

而sleep只是暂时休眠一定时间,时间到了之后,自动恢复运行,不需另外的线程唤醒.

参考代码:

using System;
using System.Collections.Generic;
using System.Text;
using System.Threading;
namespace ConsoleApplication1
{
   
    class Program
    {
        static readonly object _locker = new object();
 
        static bool _go;
        static void Main(string[] args)
        {
            new Thread(Work).Start(); //新线程会被阻塞,因为_go == false
            Console.ReadLine(); //等待用户输入
            lock (_locker)
            {
                _go = true//改变阻塞条件
                Monitor.Pulse(_locker); //通知等待的队列。
            }
            Thread.Sleep(1000);
        }
 
        static void Work()
        {
            lock (_locker)
            {
                while (!_go) //只要_go字段是false,就等待。
                    Monitor.Wait(_locker); //在等待的时候,锁已经被释放了。
            }
            Console.WriteLine("被唤醒了");
        }
        
    }
}

### C# 中 `Task.Wait` `await Task.Run` 的区别及用法 #### 使用场景差异 当处理异步编程时,`Task.Wait` 是一种阻塞调用方法,它会等待指定的任务完成执行。而 `await Task.Run` 则是非阻塞的方式,在不阻碍当前线程的情况下继续其他操作直到任务完成。 对于如下同步方法: ```csharp double ComputeNumber() { // Simulate work Thread.Sleep(5000); // Return some number return 1.0; } ``` 将其转换成异步形式可以采用 `async Task<double>` 来返回结果[^1]。此时如果要比较两种方式,则有下面的例子来说明两者的不同之处。 #### 实现对比 ##### 使用 `Task.Wait` ```csharp public void UsingTaskWait() { var task = Task.Run(() => ComputeNumber()); task.Wait(); // 阻塞主线程直至计算结束 Console.WriteLine($"Computed Number: {task.Result}"); } ``` 这段代码会在遇到 `.Wait()` 方法时暂停整个程序流程,直到后台运行的任务完成后才会继续向前推进逻辑。这种方式虽然简单直接但是并不推荐用于UI或其他需要保持响应性的环境中因为这会导致界面冻结等问题。 ##### 使用 `await Task.Run` ```csharp public async Task UsingAwaitTaskRun() { double result = await Task.Run(() => ComputeNumber()); // 不会阻塞主线程 Console.WriteLine($"Computed Number: {result}"); } ``` 相比之下,这里通过 `await` 关键字实现了更加优雅的解决方案——允许应用程序在等待长时间运算的同时还能做别的事情而不必担心卡顿现象的发生。这种方法非常适合于现代多核处理器架构下的并发处理需求。 #### 性能考量 使用 `Task.Wait` 可能在某些情况下引发死锁问题特别是当你在一个已经标记为 `[STAThread]` 或者具有特定Synchronization Context环境里尝试去阻止一个异步过程的时候;相反地,利用 `await` 结合 `Task.Run` 能够有效规避此类风险并提高整体性能表现。 综上所述,在大多数实际应用场景下建议优先选用基于 `await` 的方案来进行异步开发实践以获得更好的用户体验以及更高的效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值