在Java里如何使用Lock接口实现线程同步_Java显式锁使用方式说明

发布时间 - 2026-01-24 00:00:00    点击率:
Lock接口需手动获取和释放锁,必须在finally中调用unlock();ReentrantLock支持重入、可中断、超时等特性;tryLock()用于非阻塞或超时获取;默认非公平,公平锁性能较差;锁粒度与持有时间需设计合理。

Lock接口比synchronized更灵活,但必须手动释放

Java里用Lock实现线程同步,核心是显式获取和释放锁,不像synchronized那样自动管理。这意味着你得自己保证unlock()一定被执行,否则会永久阻塞其他线程——这是最常踩的坑。

  • 必须在finally块中调用lock.unlock(),哪怕业务逻辑抛异常也不能跳过
  • ReentrantLock是最常用的实现类,支持重入、可中断、

    超时获取等特性
  • 不能像synchronized那样直接修饰方法或代码块,必须通过对象引用调用lock()unlock()

正确写法:try-finally包裹lock/unlock

下面是最小安全模板。漏掉finally或把unlock()写在try里,都可能导致死锁。

private final Lock lock = new ReentrantLock();

public void doSomething() {
    lock.lock();
    try {
        // 临界区代码
        System.out.println("执行受保护操作");
    } finally {
        lock.unlock(); // 必须在这里,且只调用一次
    }
}
  • 不要在catch里释放锁——异常可能发生在lock()之后、try开始之前,此时unlock()会抛IllegalMonitorStateException
  • 同一个Lock实例不能被不同线程交替调用lock()/unlock(),必须成对出现在同一线程中
  • 如果需要条件等待,搭配lock.newCondition(),而不是wait()/notify()

tryLock()适合避免无限等待的场景

当线程不想一直卡在锁上,可以用tryLock()非阻塞获取,或带超时的tryLock(long, TimeUnit)。它返回boolean,成功才进临界区。

  • 返回false不代表出错,只是当前拿不到锁——你要自己决定是重试、降级还是直接返回
  • 即使tryLock()失败,也不用调用unlock();只有它返回true后才需要配对释放
  • 注意:超时时间单位必须用TimeUnit枚举,不能传毫秒整数,否则调用的是另一个重载方法(无参版本)
if (lock.tryLock(100, TimeUnit.MILLISECONDS)) {
    try {
        // 执行操作
    } finally {
        lock.unlock();
    }
} else {
    // 处理获取失败,比如记录日志或返回默认值
}

公平锁不是默认选项,性能通常更差

ReentrantLock默认是非公平锁,意味着新线程可能插队成功获取锁,而排队中的线程继续等待。开启公平模式要显式传true

Lock fairLock = new ReentrantLock(true);
  • 公平锁能减少线程饥饿,但每次获取都要遍历等待队列,吞吐量明显下降
  • 除非明确遇到“某些线程长期抢不到锁”的问题,否则别开公平模式
  • 公平性只影响等待队列中的线程顺序,不影响已持有锁的线程继续重入

真正难处理的,是锁的粒度和持有时间——锁太粗会拖慢并发,太细则容易引发死锁或重复加锁。这些没法靠Lock接口本身解决,得靠设计时厘清共享状态边界。


# java  # ai  # 有锁 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 香港服务器部署网站为何提示未备案?  深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?  Windows10如何更改计算机工作组_Win10系统属性修改Workgroup  在线ppt制作网站有哪些软件,如何把网页的内容做成ppt?  如何在腾讯云免费申请建站?  Microsoft Edge如何解决网页加载问题 Edge浏览器加载问题修复  Laravel如何生成API文档?(Swagger/OpenAPI教程)  iOS中将个别页面强制横屏其他页面竖屏  iOS正则表达式验证手机号、邮箱、身份证号等  canvas 画布在主流浏览器中的尺寸限制详细介绍  EditPlus中的正则表达式 实战(4)  Windows10电脑怎么查看硬盘通电时间_Win10使用工具检测磁盘健康  node.js报错:Cannot find module 'ejs'的解决办法  如何在宝塔面板中修改默认建站目录?  linux写shell需要注意的问题(必看)  Laravel如何使用Blade组件和插槽?(Component代码示例)  Laravel如何实现全文搜索功能?(Scout和Algolia示例)  使用spring连接及操作mongodb3.0实例  阿里云网站搭建费用解析:服务器价格与建站成本优化指南  Laravel如何使用Service Container和依赖注入?(代码示例)  湖南网站制作公司,湖南上善若水科技有限公司做什么的?  DeepSeek是免费使用的吗 DeepSeek收费模式与Pro版本功能详解  Python并发异常传播_错误处理解析【教程】  linux top下的 minerd 木马清除方法  Laravel Eloquent访问器与修改器是什么_Laravel Accessors & Mutators数据处理技巧  如何快速搭建高效香港服务器网站?  html5如何设置样式_HTML5样式设置方法与CSS应用技巧【教程】  Swift中循环语句中的转移语句 break 和 continue  如何在Windows服务器上快速搭建网站?  Windows11怎样设置电源计划_Windows11电源计划调整攻略【指南】  香港服务器网站卡顿?如何解决网络延迟与负载问题?  Laravel用户密码怎么加密_Laravel Hash门面使用教程  重庆市网站制作公司,重庆招聘网站哪个好?  Laravel DB事务怎么使用_Laravel数据库事务回滚操作  北京网站制作的公司有哪些,北京白云观官方网站?  Swift中switch语句区间和元组模式匹配  html文件怎么打开证书错误_https协议的html打开提示不安全【指南】  JavaScript如何实现路由_前端路由原理是什么  通义万相免费版怎么用_通义万相免费版使用方法详细指南【教程】  Internet Explorer官网直接进入 IE浏览器在线体验版网址  Python图片处理进阶教程_Pillow滤镜与图像增强  Laravel怎么使用Markdown渲染文档_Laravel将Markdown内容转HTML页面展示【实战】  Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】  Laravel如何实现邮件验证激活账户_Laravel内置MustVerifyEmail接口配置【步骤】  网站制作软件有哪些,制图软件有哪些?  laravel怎么用DB facade执行原生SQL查询_laravel DB facade原生SQL执行方法  如何快速上传建站程序避免常见错误?  如何在香港服务器上快速搭建免备案网站?  如何在IIS中新建站点并解决端口绑定冲突?  Laravel的辅助函数有哪些_Laravel常用Helpers函数提高开发效率