首页 > 编程语言 >Java并发(十四)----悲观互斥与乐观重试

Java并发(十四)----悲观互斥与乐观重试

时间:2023-07-27 20:24:20浏览次数:38  
标签:Java ts 重试 互斥 amount Integer balance public

1. 悲观互斥

互斥实际是悲观锁的思想

例如,有下面取款的需求

interface Account {
    // 获取余额
    Integer getBalance();
​
    // 取款
    void withdraw(Integer amount);
​
    /**
     * 方法内会启动 1000 个线程,每个线程做 -10 元 的操作
     * 如果初始余额为 10000 那么正确的结果应当是 0
     */
    static void demo(Account account) {
        List<Thread> ts = new ArrayList<>();
        for (int i = 0; i < 1000; i++) {
            ts.add(new Thread(() -> {
                account.withdraw(10);
            }));
        }
        long start = System.nanoTime();
        ts.forEach(Thread::start);
        ts.forEach(t -> {
            try {
                t.join();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        });
        long end = System.nanoTime();
        System.out.println(account.getBalance()
                + " cost: " + (end-start)/1000_000 + " ms");
    }
}

用互斥来保护

class AccountSync implements Account {
​
    private Integer balance;
​
    public AccountUnsafe(Integer balance) {
        this.balance = balance;
    }
​
    @Override
    public Integer getBalance() {
        synchronized (this) {
            return this.balance;
        }
    }
​
    @Override
    public void withdraw(Integer amount) {
        synchronized (this) {
            this.balance -= amount;
        }
    }
}

2. 乐观重试

另外一种是乐观锁思想,它其实不是互斥

class AccountCas implements Account {
    private AtomicInteger balance;
​
    public AccountCas(int balance) {
        this.balance = new AtomicInteger(balance);
    }
​
    @Override
    public Integer getBalance() {
        return balance.get();
    }
​
    @Override
    public void withdraw(Integer amount) {
        while(true) {
            // 获取余额的最新值
            int prev = balance.get();
            // 要修改的余额
            int next = prev - amount;
            // 真正修改
            if(balance.compareAndSet(prev, next)) {
                break;
            }
        }
    }
}
​
​
​
​

标签:Java,ts,重试,互斥,amount,Integer,balance,public
From: https://www.cnblogs.com/xiaoyh/p/17155363.html

相关文章

  • rabbitmq php 发送延迟消息 java 进行消费
      //-------延迟队列php需要安装好几个扩展 php生产者$connection=newAMQPStreamConnection('xxxxx',5672,'guest','guest');$channel=$connection->channel();//发送消息到交换机,并设置x-delay属性$messageData=['msg'=>json_e......
  • String转Map java
    String转Mapjava实现步骤1.理解需求在开始编写代码之前,我们需要明确我们的需求是什么。在这个任务中,我们需要将一个字符串转换为一个Java中的Map对象。字符串的格式可能是键值对的形式,比如"key1=value1;key2=value2",我们需要将其转变为一个Map对象,其中键是字符串中的键名,而值是......
  • 在Java和C#中计算SHA-1哈希
    Java版本:publicvoidtestHash(){Stringpassword="Test";byte[]key=password.getBytes();MessageDigestmd=MessageDigest.getInstance("SHA-1");byte[]hash=md.digest(key);Stringresult="";for(byteb:hash){res......
  • C#与Java互通AES算法加密解密
    C#需要引用System.Security.Cryptography命名空间///<summary>AES加密</summary>///<paramname="text">明文</param>///<paramname="key">密钥,长度为16的字符串</param>///<paramname="iv">偏移量,长度为16的字符串<......
  • 深入研究java.lang.ThreadLocal类
    Synchronized用于线程间的数据共享,而ThreadLocal则用于线程间的数据隔离。一、概述ThreadLocal是什么呢?其实ThreadLocal并非是一个线程的本地实现版本,它并不是一个Thread,而是threadlocalvariable(线程局部变量)。也许把它命名为ThreadLocalVar更加合适。线程局部变量(ThreadLocal)......
  • android studio javadoc 生成
    AndroidStudioJavadoc生成教程介绍在Android开发中,Javadoc是一种用于生成API文档的工具。通过使用Javadoc,可以为自己的代码生成详细的文档,方便团队协作和代码维护。本教程将介绍如何在AndroidStudio中生成Javadoc。流程下面是生成Javadoc的流程,你可以按照这些步骤来实现:......
  • 学习Java的第7天
    数据类型强类型语言(安全性高-速度慢)要求变量的使用要严格符合规定,所有变量都必须先定义后才能使用弱类型语言(安全性低--速度快) Java的数据类型分为两大类基本类型(primitivetype)//**八大基本数据类型**//**整数**intnum1=10;//**最常用**bytenum2=20;......
  • JavaScript中的MAP对象
    MAP对象,存储格式为键值对。和普通对象无异,不过它的方法要特殊一些。constmap=newMap();//createdmap.set(key,value);//赋值map.get(key);//读取map.delete(key);//删除指定键值map.clear();//删除所有键值对MAP对象主要用于频繁进行赋值删除时,会有更好的性能,同......
  • 运行 'Tomcat 8.5.31' 出错: 无法打开调试器端口 (127.0.0.1:62511): java.net.Socket
    多个中间件占用一个端口,修改端口  ......
  • 有关java语法的一些细节(与c++比较)
    与c++不同,java不支持方法参数的默认值设置,因此若有此需要,可以使用重载的方式去实现,如:publicclassTest{publicintfunc(inta,Stringb){//TODO}publicintfunc(){returnfunc(0,newString("defult"));}...}java中类型转换都......