64人参与 • 2026-05-09 • Asp.net
死锁是指两个或多个线程在执行过程中,由于互相等待对方持有的资源,导致所有线程都无法继续执行的状态。就像两个人面对面站在门口,谁也不肯让路,结果谁都进不了门。
死锁是指两个或多个线程相互等待对方释放资源,导致所有线程都无法继续执行。死锁的发生必须同时满足以下四个条件:
原因分析:线程a先获取lock1,线程b先获取lock2,然后互相等待对方释放锁。
// 死锁示例
object lock1 = new object();
object lock2 = new object();
// 线程a
task.run(() =>
{
lock (lock1)
{
thread.sleep(100);
lock (lock2)
{
console.writeline("thread a got both locks");
}
}
});
// 线程b
task.run(() =>
{
lock (lock2)
{
thread.sleep(100);
lock (lock1)
{
console.writeline("thread b got both locks");
}
}
});原因分析:.result会阻塞当前线程,而await需要回到ui线程继续执行,导致循环等待。
// 经典的同步上下文死锁
private async void button_click(object sender, eventargs e)
{
// 在ui线程中调用
var result = getresultasync().result; // 死锁!
label.text = result;
}
private async task<string> getresultasync()
{
await task.delay(1000);
return "result";
}原因分析:所有线程池线程都在等待某个任务完成,而该任务又需要线程池线程来执行。
// 线程池死锁示例
public void threadpooldeadlock()
{
task.run(() =>
{
task.run(() =>
{
// 内部任务需要等待外部任务完成
}).wait(); // 死锁!
}).wait();
}# 加载sos扩展 .loadby sos clr # 查看所有线程 !threads # 查看死锁情况 !dlk
// 使用monitor.tryenter设置超时
if (monitor.tryenter(lockobject, timespan.fromseconds(5)))
{
try
{
// 执行临界区代码
}
finally
{
monitor.exit(lockobject);
}
}
else
{
// 处理超时情况,可能是死锁
console.writeline("lock acquisition timed out - potential deadlock!");
}确保所有线程按照相同的顺序获取锁:
// ✅ 正确做法:统一锁获取顺序
object lock1 = new object();
object lock2 = new object();
task.run(() =>
{
// 始终按照lock1 -> lock2的顺序获取
lock (lock1)
{
lock (lock2)
{
// 安全执行
}
}
});
task.run(() =>
{
// 也按照lock1 -> lock2的顺序
lock (lock1)
{
lock (lock2)
{
// 安全执行
}
}
});使用 monitor.tryenter 设置超时时间:
// 使用monitor.tryenter避免无限等待
public bool tryacquirelock(object lockobject, timespan timeout)
{
if (monitor.tryenter(lockobject, timeout))
{
try
{
// 执行临界区代码
return true;
}
finally
{
monitor.exit(lockobject);
}
}
return false; // 超时,可能死锁
}// ✅ 正确做法:使用async/await
private async void button_click(object sender, eventargs e)
{
var result = await getresultasync();
label.text = result;
}
// 或者使用configureawait(false)
private async task<string> getresultasync()
{
await task.delay(1000).configureawait(false);
return "result";
}// 使用semaphoreslim替代多个lock
private semaphoreslim _semaphore = new semaphoreslim(1, 1);
public async task safeoperationasync()
{
await _semaphore.waitasync();
try
{
// 执行临界区代码
}
finally
{
_semaphore.release();
}
}
// 使用readerwriterlockslim实现读写分离
private readerwriterlockslim _rwlock = new readerwriterlockslim();
public void readoperation()
{
_rwlock.enterreadlock();
try
{
// 读取操作
}
finally
{
_rwlock.exitreadlock();
}
}// ✅ 正确做法
public async task processdataasync()
{
var task1 = getdata1async();
var task2 = getdata2async();
await task.whenall(task1, task2);
var result1 = await task1;
var result2 = await task2;
// 处理结果
}// ❌ 错误做法
lock (lockobject)
{
var data = getdatafromdatabase(); // 耗时操作
processdata(data);
}
// ✅ 正确做法
var data = getdatafromdatabase(); // 先获取数据
lock (lockobject)
{
processdata(data); // 只锁定必要的代码
}// ✅ 推荐使用lock语句(自动处理异常情况)
lock (lockobject)
{
// 临界区代码
}
// 而不是手动使用monitor
monitor.enter(lockobject);
try
{
// 临界区代码
}
finally
{
monitor.exit(lockobject);
}// ❌ 危险做法
lock (lockobject)
{
callback(); // 外部回调可能持有其他锁
}
// ✅ 安全做法
var localdata = getdata();
callback(localdata); // 在锁外调用
lock (lockobject)
{
updatestate(localdata);
}// 实现异步锁
public class asynclock
{
private readonly semaphoreslim _semaphore = new semaphoreslim(1, 1);
private readonly task<idisposable> _releaser;
public asynclock()
{
_releaser = task.fromresult((idisposable)new releaser(this));
}
public task<idisposable> lockasync()
{
var wait = _semaphore.waitasync();
return wait.iscompleted ?
_releaser :
wait.continuewith((_, state) => (idisposable)state,
_releaser.result, taskscheduler.default);
}
private sealed class releaser : idisposable
{
private readonly asynclock _lock;
public releaser(asynclock @lock) => _lock = @lock;
public void dispose() => _lock._semaphore.release();
}
}
// 使用示例
private readonly asynclock _asynclock = new asynclock();
public async task safeasyncoperation()
{
using (await _asynclock.lockasync())
{
// 异步临界区代码
}
}表格
| 策略 | 适用场景 | 优点 | 缺点 |
|---|---|---|---|
| 锁顺序一致性 | 多锁场景 | 简单有效 | 需要全局协调 |
| 锁超时机制 | 不确定等待时间 | 可检测死锁 | 可能误判 |
| 异步编程 | ui应用、i/o操作 | 避免线程阻塞 | 代码复杂度增加 |
| 高级同步原语 | 复杂并发场景 | 灵活性高 | 学习成本高 |
| 无锁编程 | 高性能要求 | 最高性能 | 实现复杂,易出错 |
避免死锁的关键在于打破死锁的四个必要条件之一:
通过合理的设计和编码实践,可以有效预防和解决c#中的死锁问题,提高程序的稳定性和性能。
以上就是c#死锁发生原因与优化解决方案的详细内容,更多关于c#死锁原因及解决的资料请关注代码网其它相关文章!
您想发表意见!!点此发布评论
版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。
发表评论