ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

初识同步锁

2020-12-14 17:01:48  阅读:138  来源: 互联网

标签:同步 synchronized lock 读锁 初识 线程 Lock static


我们知道,锁是用来控制多个线程访问共享资源的方式,一般来说,一个锁能够防止多个线程同时访问共享资源,在Lock接口出现之前,Java应用程序只能依靠synchronized关键字来实现同步锁的功能,在java5以后,增加了JUC

的并发包且提供了Lock接口用来实现锁的功能,它提供了与synchroinzed关键字类似的同步功能,只是它比synchronized更灵活,能够显示的获取和释放锁。

Lock的初步使用

Lock是一个接口


两个核心的方法lock和unlock。有6个实现类


ReentrantLock

重入锁,表示支持重新进入的锁,也就是说,如果当前线程t1通过调用lock方法获取了锁之后,再次调用lock,是不会再阻塞去获取锁的,直接增加重试次数就行了。

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

/**
 * @auther: tianweichang
 * @date: 2018/9/5 06
 * @Description:
 */
public class AtomicDemo {
private static int count=0;

    static Lock lock=new ReentrantLock();
    private static void inc(){
lock.lock();
        try {
            Thread.sleep(1);
            System.out.println(count);
        }catch (InterruptedException e){
            e.printStackTrace();
        }
count++;
        lock.unlock();
    }

public static void main(String[] args) {
for (int i=0;i<1000;i++){
new Thread(()->{AtomicDemo.inc();}).start();
        }
try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("count="+count);
    }
}

ReentrantReadWriteLock

我们以前理解的锁,基本都是排他锁,也就是这些锁在同一时刻只允许一个线程进行访问,而读写所在同一时刻可以允许多个线程访问,但是在写线程访问时,所有的读线程和其他写线程都会被阻塞。读写锁维护了一对锁,一个读锁、一个写锁; 一般情况下,读写锁的性能都会比排它锁好,因为大多数场景读是多于写的。在读多于写的情况下,读写锁能够提供比排它锁更好的并发性和吞吐量

import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantReadWriteLock;

/**
 * @auther: tianweichang
 * @date: 2018/9/5 06
 * @Description:
 */
public class LockDemo {
static Map<String, Object> cacheMap = new HashMap<>();
    static ReentrantReadWriteLock rwl = new ReentrantReadWriteLock();
    static Lock read = rwl.readLock();//读锁
    static Lock write = rwl.writeLock();//写锁

    public static final Object get(String key) {
        System.out.println("开始读取数据");
        read.lock(); //读锁
        try {
return cacheMap.get(key);
        } finally {
read.unlock();
        }
    }

public static final Object put(String key, Object value) {
write.lock();
        System.out.println("开始写数据");
        try {
return cacheMap.put(key, value);
        } finally {
write.unlock();
        }
    }
}

在这个案例中,通过hashmap来模拟了一个内存缓存,然后使用读写所来保证这个内存缓存的线程安全性。当执行读操作的时候,需要获取读锁,在并发访问的时候,读锁不会被阻塞,因为读操作不会影响执行结果。在执行写操作是,线程必须要获取写锁,当已经有线程持有写锁的情况下,当前线程会被阻塞,只有当写锁释放以后,其他读写操作才能继续执行。使用读写锁提升读操作的并发性,也保证每次写操作对所有的读写操作的可见性

  • 读锁与读锁可以共享

  • 读锁与写锁不可以共享(排他)

  • 写锁与写锁不可以共享(排他)


Lock和synchronized的简单对比

通过我们对Lock的使用以及对synchronized的了解,基本上可以对比出这两种锁的区别了。因为这个也是在面试过程中比较常见的问题

  1. 从层次上,一个是关键字、一个是类, 这是最直观的差异

  2. 从使用上,lock具备更大的灵活性,可以控制锁的释放和获取; 而synchronized的锁的释放是被动的,当出现异常或者同步代码块执行完以后,才会释放锁

  3.  lock可以判断锁的状态、而synchronized无法做到


lock可以实现公平锁、非公平锁; 而synchronized只有非公平锁


标签:同步,synchronized,lock,读锁,初识,线程,Lock,static
来源: https://blog.51cto.com/10983206/2564042

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有