redis锁实现_IDEA使用try

redis锁实现_IDEA使用tryWeb全栈~35.显式锁上一期接口Lock显式锁接口的定义       lock()/unlock():就是普通的获取锁和释放锁方法,lock()会阻塞直到成功。       lockInterruptibly():与lock()的不同是,它可以响应中断,如果被其他线程中断了,则抛出InterruptedException。  &nbsp

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

Web全栈~35.显式锁

上一期

接口Lock

显式锁接口的定义

在这里插入图片描述

       lock()/unlock():就是普通的获取锁和释放锁方法,lock()会阻塞直到成功。

       lockInterruptibly():与lock()的不同是,它可以响应中断,如果被其他线程中断了,则抛出InterruptedException。

       tryLock():只是尝试获取锁,立即返回,不阻塞,如果获取成功,返回true,否则返回false。

       tryLock(long time,TimeUnit unit):先尝试获取锁,如果能成功则立即返回true,否则阻塞等待,但等待的最长时间由指定的参数设置,在等待的同时响应中断,如果发生了中断,抛出InterruptedException,如果在等待的时间内获得了锁,返回true,否则返回false。

       newCondition:新建一个条件,一个Lock可以关联多个条件

       可以看出,相比synchronized,显式锁支持以非阻塞方式获取锁、可以响应中断、可以限时,这使得它灵活得多。

ReentrantLock

       Lock接口的主要实现类是ReentrantLock,它的基本用法lock/unlock实现了与synchronized一样的语义

       ReentrantLock有两个构造方法

在这里插入图片描述

       参数fair表示是否保证公平,不指定的情况下,默认为false,表示不保证公平。所谓公平是指,等待时间最长的线程优先获得锁。保证公平会影响性能,一般也不需要,所以默认不保证,synchronized锁也是不保证公平的

       使用显式锁,一定要记得调用unlock。一般而言,应该将lock之后的代码包装到try语句内,在finally语句内释放锁。

代码示例

 public void fun() { 
   
      lock.lock();
      try{ 
   
          count++;
      }finally { 
   
          lock.unlock();
      }
  } 

使用tryLock()避免死锁

       使用tryLock(),可以避免死锁。在持有一个锁获取另一个锁而获取不到的时候,可以释放已持有的锁,给其他线程获取锁的机会,然后重试获取所有锁。

银行转账实例

账户类
class Account{ 
   
    private Lock lock = new ReentrantLock();
    private volatile double money;
    public Account(double initialMoney){ 
   
        this.money = initialMoney;
    }
    public void add(double money){ 
   
        lock.lock();
        try{ 
   
            this.money = money;
        }finally { 
   
            lock.unlock();
        }
    }
    public void reduce(double money){ 
   
        lock.lock();
        try{ 
   
            this.money -= money;
        }finally { 
   
            lock.unlock();
        }
    }
    public double getMoney() { 
   
        return money;
    }
    public void lock(){ 
   
        lock.lock();
    }
    public void unlock(){ 
   
        lock.unlock();
    }
    public boolean tryLock(){ 
   
        return lock.tryLock();
    }
}

       在账户之间转账,需要两个账户都锁定,如果不使用tryLock,而直接使用lock就会发生死锁。这里可以试一试

class NoEnoughMoney extends Exception { 
   
    public static void transfer(Account from,Account to,double money)
        throws NoEnoughMoney { 
   
        from.lock();
        try { 
   
            to.lock();
            try { 
   
            if(from.getMoney() >= money) { 
   
                from.reduce(money);
                to.add(money);
            } else { 
   
                throw new NoEnoughMoney();
            }
            } finally { 
   
                to.unlock();
            }
        } finally { 
   
            from.unlock();
        }
    }
}
模拟死锁的过程
public static void simulateDeadLock() { 
   
    final int accountNum = 10;
    final Account[]accounts = new Account[accountNum];
    final Random rnd = new Random();
    for(int i = 0; i < accountNum; i++) { 
   
        accounts[i]= new Account(rnd.nextInt(10000));
    }
    int threadNum = 100;
    Thread[]threads = new Thread[threadNum];
    for(int i = 0; i < threadNum; i++) { 
   
        threads[i]= new Thread() { 
   
            public void run() { 
   
                int loopNum = 100;
                for(int k = 0; k < loopNum; k++) { 
   
                    int i = rnd.nextInt(accountNum);
                    int j = rnd.nextInt(accountNum);
                    int money = rnd.nextInt(10);
                    if(i!= j) { 
   
                        try { 
   
                            transfer(accounts[i],accounts[j],money);
                        } catch (NoEnoughMoney e) { 
   
                        }
                    }
                }
            }
        };
        threads[i].start();
    }
}

       这里有10个账户,100个线程,每个线程执行100次循环,在每次循环中,随机挑选两个账户进行转账。每次执行该段代码都会发生死锁。

使用tryLock实现转账

public static boolean tryTransfer(Account from,Account to,double money) throws NoEnoughMoney { 
   
        if(from.tryLock()) { 
   
            try { 
   
                if(to.tryLock()) { 
   
                    try { 
   
                        if(from.getMoney() >= money) { 
   
                            from.reduce(money);
                            to.add(money);
                        } else { 
   
                            throw new NoEnoughMoney();
                        }
                        return true;
                    } finally { 
   
                        to.unlock();
                    }
                }
            } finally { 
   
                from.unlock();
            }
        }
        return false;
    }

       如果两个锁都能够获得,且转账成功,则返回true,否则返回false。不管怎样,结束都会释放所有锁。

       transfer方法可以循环调用该方法以避免死锁

public static void transfer(Account from,Account to,double money) throws NoEnoughMoney { 
   
    boolean success = false;
    do { 
   
        success = tryTransfer(from,to,money);
        if(!success) { 
   
            Thread.yield();
        }
    } while (!success);
}
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/181684.html原文链接:https://javaforall.net

(0)
上一篇 2022年10月15日 下午3:36
下一篇 2022年10月15日 下午3:46


相关推荐

  • 凸包算法(convex hull)

    凸包算法(convex hull)前言今天学习了几何算法 凸包算法 听着名字很奇怪 不知道它是干什么的 但其实也很简单 下面来介绍一下这种凸包算法和其 c 代码 凸包算法其实很简单 就是用一个的凸多边形围住所有的点 就好像桌面上有许多图钉 用一根紧绷的橡皮筋将它们全部围起来一样 算法详细步骤 1 找到所有点中纵坐标 y 最小的点 也就是这些点中最下面的点 记为 p0 2 然后计算其余点与该点的连线与 x 轴之间夹角

    2026年3月17日
    1
  • Java多态性:Java什么是多态?

    Java多态性:Java什么是多态?多态性是面向对象编程的又一个重要特征,它是指在父类中定义的属性和方法被子类继承之后,可以具有不同的数据类型或表现出不同的行为,这使得同一个属性或方法在父类及其各个子类中具有不同的含义。对面向对象来说,多态分为编译时多态和运行时多态。其中编译时多态是静态的,主要是指方法的重载,它是根据参数列表的不同来区分不同的方法。通过编译之后会变成两个不同的方法,在运行时谈不上多态。而运行时多态是动态的,它是通…

    2022年7月8日
    23
  • linux运行gradle,Linux Gradle 安装

    linux运行gradle,Linux Gradle 安装安装步骤下载 gradle 安装包解压安装包设置环境变量可以到官网 查看具体 install 详情 1 下载 gradle 安装包 wgethttps downloads gradle org distribution gradle 4 4 1 all zip2 解压安装包注 若没 unzip 命令 则记得安装 yuminstall yunzipzip 开始解压 unzpgradle 4 4 1

    2026年3月16日
    2
  • 如何自己搭建服务器_文件服务器搭建

    如何自己搭建服务器_文件服务器搭建FlashFXP使用其实就是为了实现文件的上传和下载,它要结合

    2025年10月25日
    5
  • bcdboot(引导修复工具) 命令行工具使用方法

    bcdboot(引导修复工具) 命令行工具使用方法BCDboot 是一种用于快速设置系统分区或修复系统分区上的启动环境的工具 系统分区是通过从已安装的 Windows R 映像复制一小部分启动环境文件来设置的 BCDboot 还会在系统分区上创建引导配置数据 BCD 存储 该存储中包含一个新的引导项 可让您引导到已安装的 Windows 映像 软件名称 bcdboot exe 引导修复工具软件大小 68KB 更新时间 2015 08 06

    2026年3月17日
    2
  • WatchDog+Redis实现分布式锁

    WatchDog+Redis实现分布式锁代码说明 watchDog 机制主要是用来对 redis 中商品 key 进行锁续命 防止业务处理时间过长导致的误删 key 值 lua 脚本则用来对 redis 中指令的原子性操作 注意 lua 脚本中不能有复杂逻辑 防止阻塞 redis 创建定时任务线程工厂 privatestati FACTORY newThreadFac setNameForma watchDog get

    2026年3月20日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号