8、synchronized和Lock的使用
千里之行,始于足下
正文
1、多并发案例:
一个车站有三个窗口同时卖30张票,每个窗口都有40个人在排队买票,在多线程情况下,不加锁,线程不安全,导致卖票不准确
package com.example.Lock; /** * 线程就是一个单独的资源类,没有任何附属的操作 */ public class demo1 { public static void main( String[] args ) { //并发:多线程操作同一个资源类,把资源类丢入线程 Ticket ticket = new Ticket(); new Thread(() -> { for(int i=0;i<40;i++){ ticket.sale(); } },"A窗口").start(); new Thread(() -> { for(int i=0;i<40;i++){ ticket.sale(); } },"B窗口").start(); new Thread(() -> { for(int i=0;i<40;i++){ ticket.sale(); } },"c窗口").start(); } } /** * 资源类 */ class Ticket{ //总票数 private int number = 30; //卖票 public void sale(){ if(number>0){ System.out.println(Thread.currentThread().getName()+"卖出了第"+(number--)+"张票,剩余: "+number); } } }
2、使用锁:
解决上面的线程不安全的问题,可以通过加锁的方式
(1)、方式一:synchronized关键字
使用synchronized,在卖票的方法加synchronized关键字,表示对是整个方法范围内对当前对象的加锁
//卖票 public synchronized void sale(){ if(number>0){ System.out.println(Thread.currentThread().getName()+"卖出了第"+(number--)+"张票,剩余: "+number); } }
(2)、方式二:Lock接口
使用Lock,一共三步
——》创建锁
——》加锁
——》解锁
/** * 资源类 */ class Ticket{ //总票数 private int number = 30; //创建锁 Lock lock = new ReentrantLock(); //卖票 public void sale(){ //加锁 lock.lock(); try { if(number>0){ System.out.println(Thread.currentThread().getName()+"卖出了第"+(number--)+"张票,剩余: "+number); } } catch (Exception e) { }finally{ //解锁 lock.unlock(); } } }
3、synchronized和Lock的区别:
(1)、synchronized是内置的Java关键字;Lock是一个接口
(2)、synchronized无法判断获取锁的状态;Lock可以
(3)、synchronized会自动释放锁;Lock必须手动释放锁,如果不释放,就会出现死锁
(4)、synchronized多个线程获取同一个锁,其中一个线程已经获取锁,但发生阻塞,其他线程就会一直等待;Lock不一定等待下去
(5)、synchronized是可重入锁,不可以中断的,非公平;Lock也是可重入锁,可以判断锁,默认非公平(可以设置公平)
(6)、synchronized适合锁少量代码同步;Lock适合锁大量同步代码
分类:
3、Java技巧实例
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· winform 绘制太阳,地球,月球 运作规律
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人