关于java:Java线程同步的四种方式详解建议收藏

 

Java线程同步属于Java多线程与并发编程的外围点,须要重点把握,上面我就来详解Java线程同步的4种次要的实现形式@mikechen

目录

  • 什么是线程同步
  • 线程同步的几种形式

    • 1、应用synchronized关键字
    • 2.应用ReentrantLock
    • 3.应用原子变量实现线程同步
    • 4.ThreadLocal实现线程同步

什么是线程同步

当应用多个线程来拜访同一个数据时,将会导致数据不精确,相互之间产生抵触,非常容易呈现线程平安问题,如下图所示:

比方多个线程都在操作同一数据,都打算批改商品库存,这样就会导致数据不统一的问题。

线程同步的实在意思,其实是“排队”:几个线程之间要排队,一个一个对共享资源进行操作,而不是同时进行操作。

所以咱们用同步机制来解决这些问题,退出同步锁以防止在该线程没有实现操作之前,被其余线程的调用,从而保障了该变量的唯一性和准确性。

 

线程同步的几种形式

1、应用synchronized关键字

这种形式比拟灵便,润饰一个代码块,被润饰的代码块称为同步语句块。

其作用的范畴是大括号{}括起来的代码,作用的对象是调用这个代码块的对象,如下格局:

synchronized(对象) {                    
   //失去对象的锁,能力操作同步代码
    须要被同步代码;
}

![]()

通常没有必要同步整个办法,应用synchronized代码块同步要害代码即可。

具体的示例如下:

public class SynchronizedThread {
 
    class Bank {
 
        private int account = 200;
 
        public int getAccount() {
            return account;
        }
 
        /**
         * 用同步办法实现
         *
         * @param money
         */
        public synchronized void save(int money) {
            account += money;
        }
 
        /**
         * 用同步代码块实现
         *
         * @param money
         */
        public void save1(int money) {
            synchronized (this) {
                account += money;
            }
        }
    }
 
    class NewThread implements Runnable {
        private Bank bank;
 
        public NewThread(Bank bank) {
            this.bank = bank;
        }
 
        @Override
        public void run() {
            for (int i = 0; i < 10; i++) {
                // bank.save1(10);
                bank.save(10);
                System.out.println(i + "账户余额为:" + bank.getAccount());
            }
        }
 
    }
 
    /**
     * 建设线程,调用外部类
     */
    public void useThread() {
        Bank bank = new Bank();
        NewThread new_thread = new NewThread(bank);
        System.out.println("线程1");
        Thread thread1 = new Thread(new_thread);
        thread1.start();
        System.out.println("线程2");
        Thread thread2 = new Thread(new_thread);
        thread2.start();
    }
 
    public static void main(String[] args) {
        SynchronizedThread st = new SynchronizedThread();
        st.useThread();
    }
 
}

![]()

 

2.应用ReentrantLock

ReentrantLock类是可重入、互斥、实现了Lock接口的锁,它与应用synchronized办法具备雷同的根本行为和语义,并且扩大了其能力。

private int account = 100;
            //须要申明这个锁
            private Lock lock = new ReentrantLock();
            public int getAccount() {
                return account;
            }
            //这里不再须要synchronized 
            public void save(int money) {
                lock.lock();
                try{
                    account += money;
                }finally{
                    lock.unlock();
                }
 
            }
        }

![]()

synchronized 与 Lock 的比照

ReentrantLock是显示锁,手动开启和敞开锁,别忘记敞开锁;

synchronized 是隐式锁,出了作用域主动开释;

ReentrantLock只有代码块锁,synchronized 有代码块锁和办法锁;

应用 ReentrantLock锁,JVM 将破费较少的工夫来调度线程,线程更好,并且具备更好的扩展性(提供更多的子类);

优先应用程序:

ReentrantLock> synchronized 同步代码块> synchronized 同步办法

 

3.应用原子变量实现线程同步

为了实现线程同步,咱们将应用原子变量(Atomic*结尾的)来实现。

比方典型代表:AtomicInteger类存在于java.util.concurrent.atomic中,该类示意反对原子操作的整数,采纳getAndIncrement办法以原子办法将以后的值递减。

具体示例如下:

private AtomicInteger account = new AtomicInteger(100);
 
        public AtomicInteger getAccount() {
            return account;
        }
 
        public void save(int money) {
            account.addAndGet(money);
        }

![]()

4.ThreadLocal实现线程同步

如果应用ThreadLocal治理变量,则每一个应用该变量的线程都取得该变量的正本,正本之间互相独立,这样每一个线程都能够随便批改本人的变量正本,而不会对其余线程产生影响,从而实现线程同步。

具体代码示例如下:

//只改Bank类,其余代码与上同
        public class Bank{
            // 创立一个线程本地变量 ThreadLocal
            private static ThreadLocal<Integer> account = new ThreadLocal<Integer>(){
                @Override
                //返回以后线程的"初始值" 
                protected Integer initialValue(){
                    return 100;
                }
            };
            public void save(int money){
                //设置线程正本中的值
                account.set(account.get()+money);
            }
            public int getAccount(){
                //返回线程正本中的值 
                return account.get();
            }
        }

![]()

以上

作者简介

陈睿|mikechen,10年+大厂架构教训,《BAT架构技术500期》系列文章作者,专一于互联网架构技术。

浏览mikechen的互联网架构更多技术文章合集

Java并发|JVM|MySQL|Spring|Redis|分布式|高并发

关注「mikechen 的互联网架构」公众号,回复 【架构】 支付《Java进阶架构思维导图&Java进阶架构文章合集》

评论

发表回复

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

这个站点使用 Akismet 来减少垃圾评论。了解你的评论数据如何被处理