Lock锁接口实现

Lock锁接口实现

学习材料来源于网络
如有侵权,联系删除

源码

package java.util.concurrent.locks;
import java.util.concurrent.TimeUnit;

/**
 * 与使用{@code sync}方法和语句相比,{@ code Lock}实现提供了更广泛的锁定
 * 操作。它们允许更灵活的结构,可能具有完全不同的属性,并且可能支持多个关联的{@link Condition}对象。
 *
 * 锁是用于控制多个线程对共享资源的访问的工具。通常,锁提供对共享资源的独占访问:
 * 一次只能有一个线程可以获取该锁,并且对共享资源的所有访问都需要首先获取该锁。
 * 但是,某些锁可能允许并发访问共享资源,例如{@link ReadWriteLock}的读取锁。
 *
 * <p>使用{@code Synchronized}方法或语句可
 * 访问与每个对象关联的隐式监视器锁,但是
 * 强制所有锁的获取和释放以块结构方式进行:
 * 当多个锁被使用时被获取的锁必须以相反的
 * 顺序释放,并且所有锁都必须在与被获取的锁相同的词法范围内释放。
 *
 * <p>虽然{@code sync}方法的作用域机制
 * 和语句使使用监视器锁的编程变得容易得多,并且
 * 并避免了很多常见的涉及锁的编程错误,但是
 * 在某些情况下您需要使用锁以更灵活的方式。例如,某些用于遍历并发访问的数据结构的算法需要使用“交接”或“链锁”:您
 * 取得节点A的锁,然后取得节点B,然后释放A并取得
 * C ,然后释放B并获取D,依此类推。 
 * {@code Lock}接口的实现通过
 * 允许在不同范围内获取和释放锁,并允许以任意
 * 顺序获取和释放多个锁,从而启用了此类技术。
 *
 * <p>随着灵活性的提高,额外的责任也随之增加。缺少块结构锁定将消除
 * {{code sync}方法和语句中发生的锁定的自动释放。在大多数情况下,应使用以下习惯用语
 * :
 *
 *  <pre> {@code
 * Lock l = ...;
 * l.lock();
 * try {
 *   // access the resource protected by this lock
 * } finally {
 *   l.unlock();
 * }}</pre>
 *
 *当锁定和解锁发生在不同的范围内时,
 *必须小心以确保通过try-finally或try-catch保护在保持锁的同时执行的所有代码,
 *以确保在必要时释放锁。
 *
 * <p> {@ code Lock}类还可以提供与隐式监视器锁完全不同的行为和语义,
 * 例如,保证顺序,不可重用或死锁。如果实现提供了这种特殊的语义
 *,那么实现必须记录这些语义。
 *
 * <p> {@ code Lock}类还可以提供与隐式监视器锁完全不同的行为和语义,
 * 例如,保证顺序,不可重用或死锁。如果实现提供了这种特殊的语义,
 * 那么实现必须记录这些语义。
 *
 * <p>请注意,{@ code Lock}实例只是普通对象,它们自身可以用作{@code Synchronized}语句中的目标。 
 * 获取{@code Lock}实例的监视器锁定与该实例的任何{@link #lock}方法都没有指定的关系
 *。 
 * 为避免混淆,建议您不要以这种方式使用{@code Lock} 
 * 实例,除非在它们自己的实现中使用。
 *
 * <p>除非另有说明,否则为任何
 * 参数传递{@code null}值将导致抛出{@link NullPointerException}。
 *
 * <h3>内存同步</h3>
 *
 * <p>所有{@code Lock}实现<em>必须</ em>强制执行与内置监视器锁相同的
 * 内存同步语义,如
 * <a href =“ https:// docs.oracle.com/javase/specs/jls/se7/html/jls-17.html#jls-17.4“> 
 * Java语言规范(17.4内存模型)</a>:
 * <ul> 
 * <li>成功的{@code lock}操作具有与成功的<em> Lock </ em>操作相同的内存*同步效果。 
 * <li>成功的{@code unlock}操作与成功的<em> Unlock </ em>操作具有相同的内存同步效果。 
 * </ ul> 
 * 
 * 不成功的锁定和解锁操作,以及可重入的
 * 锁定/解锁操作,不需要任何内存*同步效果。
 *
 * <h3>实施注意事项s</h3>
 *
 * <p>锁获取的三种形式(可中断,
 * 不可中断和定时)在性能,特性,订购保证或其他实现质量上可能有所不同。
 * 此外,在给定的{@code Lock} 
 * 类中,中断<em> inginging </ em> *获取锁的功能可能不可用。
 * 因此,不需要实现为所有三种形式的锁获取定义完全相同的保证或语义,
 * 也不需要支持中断正在进行的锁获取。需要一个实现来清楚地*记录每个锁定方法所提供的语义和保证。
 * 在支持锁获取中断的程度上,它还必须遵守此接口中定义的中断语义:或者完全或仅在方法输入时才这样做。
 *
 * <p>由于中断通常意味着取消,并且对中断的检查通常是很少的,因此与正常方法返回相比,实现可能更喜欢对中断做出响应。即使可以
 * 表明在另一个操作之后发生的中断可能已经取消阻塞了线程,也是如此。实现应记录此行为。
 *
 * @see ReentrantLock
 * @see Condition
 * @see ReadWriteLock
 *
 * @since 1.5
 * @author Doug Lea
 */
public interface Lock {

	/ ** 
        *获取锁。 
        * 
        * <p>如果该锁不可用,则出于线程调度目的,当前线程将被禁用
        *,并且在获取该锁之前,该线程处于休眠状态。 
        * 
        * <p> <b>实现注意事项</ b> 
        * 
        * <p> {@ code Lock}实现可能能够检测对锁的错误使用,例如可能导致死锁的调用,并且
        * 在这种情况下抛出一个(未经检查的)异常。 *环境和异常类型必须由该{@code Lock}实现来记录。 
        * /
    void lock();

    /**
	 *获取锁定,除非当前线程被{{@linkplain Thread#interrupt interrupted}破坏。 
	 * 
	 * <p>获取锁(如果有)并立即返回。 
	 * 
	 * <p>如果该锁不可用,则出于线程调度目的,当前线程将被禁用*并处于休眠状态,直到发生以下两种情况之一:
     *
	 * <ul> 
	 * <li>该锁由当前线程获取;或
	 * <li>当前线程有一些其他线程{@linkplain Thread#interrupt interrupts},并且支持中断获取锁。 
	 * </ ul>
     *
	 * <p>如果当前线程:
	 * <ul> 
	 * <li>在进入此方法时已设置其中断状态;或
	 * <li>在获取
	 * 锁的过程中被{@linkplain Thread#interrupt interrupted中断},并且支持中断获取锁,
	 * </ ul> *然后抛出{@link InterruptedException}并且当前线程的
	 * 中断状态为已清除。
     *
	 * <p> <b>实施注意事项</ b>
     *
	 * <p>在某些*实现中可能无法中断锁获取,并且如果可能的话,可能是
	 * 昂贵的操作。程序员应意识到可能是这种情况。在这种情况下,实现应记录在案。 
	 * 
	 * <p>与正常方法返回相比,实现可能更喜欢对中断做出响应。
	 * 
	 * <p> {@ code Lock}实现可能能够检测到*错误的使用锁,例如可能导致死锁的调用,
	 * 并且在这种情况下可能引发(未经检查的)异常。环境和异常类型必须*由该{@code Lock}实现来记录。 
	 * 
	 * @throws InterruptedException如果当前线程在获取锁时被中断(并且支持锁获取的中断)
	 */
    void lockInterruptibly() throws InterruptedException;

    /**
     * 仅在调用时释放锁时才获取锁。
     *
     * <p>获取该锁(如果有)并立即返回
     *,其值为{@code true}。
     * 如果锁不可用,则此方法将立即返回
     *,其值为{@code false}。
     *
     * <p>此方法的典型用法是:
     *  <pre> {@code
     * Lock lock = ...;
     * if (lock.tryLock()) {
     *   try {
     *     // 操纵保护状态
     *   } finally {
     *     lock.unlock();
     *   }
     * } else {
     *   // 执行替代动作
     * }}</pre>
     *
     * 此用法可确保在获取锁后将其解锁,并且
     *在未获取锁时不会尝试解锁。
     *
     * @return {@code true} 如果获得了锁,并且
     *         {@code false} 否则
     */
    boolean tryLock();

    /**
     *如果锁在给定的等待时间内是空闲的,并且
     *当前线程尚未{@linkplain Thread#interrupt interrupted},则获取该锁。
     *
     * <p>如果锁可用,则此方法立即返回
     *,其值为{@code true}。
     * 如果该锁不可用,则*当前线程将出于线程调度目的而被禁用
     * 并处于休眠状态,直到发生以下三种情况之一:
     * <ul>
     * <li>该锁是由当前线程获取的;要么
     * <li>当前线程的其他一些线程
     * {@linkplain Thread#interrupt interrupts},并支持锁定获取的中断;要么
     * <li>经过指定的等待时间
     * </ul>
     *
     * <p>如果获得了锁,则返回值{@code true}。
     *
     * <p>如果当前线程:
     * <ul>
     * <li>在进入此方法时已设置其中断状态;要么
     * <li>在获取*锁的过程中被{@linkplain Thread#interrupt interrupted中断了,并且支持中断获取锁,
     * </ul>
     * 然后抛出{@link InterruptedException}并清除当前线程的
     * 中断状态。
     *
     * <p>如果经过了指定的等待时间,则返回值{@code false}
     *。
     *如果时间*小于或等于零,则该方法将完全不等待。
     *
     * <p><b>实施注意事项</b>
     *
     * <p>在某些实现中,中断锁获取的能力*可能是不可能的,并且如果可能的话
     * 是一项昂贵的操作。 *程序员应意识到可能是这种情况。在这种情况下,
     * 实现应记录在案。
     *
     * <p>与正常方法返回或报告超时相比,实现可能更喜欢响应中断。
     *
     * <p> {@ code Lock}实现可以检测*错误使用锁,例如可能导致死锁的调用,并且在这种情况下可能引发(未经检查的)异常。
     * 必须通过该{@code Lock}实现来记录情况和异常类型。
     *
     * @param time等待锁的最长时间
     * @param {@code time}参数的时间单位
     * @return {@code true},如果已获得锁,则{@code false}
     *         获取锁之前是否经过了等待时间
     *
     * @throws InterruptedException-如果当前线程在获取锁时被中断
     *(并且支持锁的中断
     * 支持获取)
     */
    boolean tryLock(long time, TimeUnit unit) throws InterruptedException;

    /**
     * 释放锁。
     *
     * <p><b>实施注意事项</b>
     *
     * <p>{@code Lock}实现通常会对可以释放锁的线程施加
     *限制(通常只有
     *
     *锁的持有者可以释放该锁),并且如果违反该限制,则可能引发
     *(未经检查)异常。
     *任何限制和例外
     *类型必须由该{@code Lock}实现来记录。
     */
    void unlock();

    /**
     *返回绑定到该
     * {@code Lock}实例的新{@link Condition}实例。
     *
     * <p>在等待条件之前,锁必须由当前线程持有。
     *对{@link Condition#await()}的调用将在等待之前自动释放锁
     *,并在等待返回之前重新获取该锁。
     *
     * <p><b>实施注意事项</b>
     *
     * <p> {@ link Condition}实例的确切操作取决于{@code Lock}实现,并且必须由该实现记录下来。
     *
     * @return 此{@code Lock}实例的新{@link Condition}实例
     * @throws UnsupportedOperationException如果此{@code Lock}
     *         实施不支持条件
     */
    Condition newCondition();
}

核心API

方法 描述
lock 获取锁的方法,若被其他线程占用,则会等待(阻塞)
lockInterruptibly 在锁的获取过程中可以中断当前线程
tryLock 尝试非阻塞地获取锁,立即返回
unlocj 释放锁

提示:根据Lock接口的源码注释,Lock接口的实现,具备和同步关键字同样的内存语义。

ReentrantLock

独享锁;支持公平锁、非公平锁两种模式;可重入

示例1

// 3、 演示可重入
public class ReentrantDemo1 {
    private static final ReentrantLock lock = new ReentrantLock();

    public static void main(String[] args) {
        //第一次加锁
        lock.lock();
        try {
            System.out.println("第一次获取锁");
            System.out.println("当前线程获取锁的次数" + lock.getHoldCount());
            //第二次加锁
            lock.lock();
            System.out.println("第二次获取锁了");
            System.out.println("当前线程获取锁的次数" + lock.getHoldCount());
        }finally {
            //释放第二次锁
            lock.unlock();
            //释放第一次锁
            lock.unlock();
        }
        System.out.println("当前线程获取锁的次数" + lock.getHoldCount());

        // 如果不释放,此时其他线程是拿不到锁的
        new Thread(() -> {
            System.out.println(Thread.currentThread() + " 期望抢到锁");
            lock.lock();
            System.out.println(Thread.currentThread() + " 线程拿到了锁");
        }).start();


    }
}

正确的停止等待锁的线程

示例2

package icu.shaoyayu.multithreading.chapter4;

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

/**
 * @author shaoyayu
 * @E_Mail
 * @Version 1.0.0
 * @readme :
 */
// ReentrantLock 可重入锁示例
public class ReentrantLockDemo1 {
    private Lock lock = new ReentrantLock();

    public static void main(String[] args) throws InterruptedException {
        ReentrantLockDemo1 demo1 = new ReentrantLockDemo1();
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                try {
                    demo1.test(Thread.currentThread());
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };
        Thread thread1 = new Thread(runnable);
        Thread thread2 = new Thread(runnable);
        thread1.start();
        Thread.sleep(500); // 等待0.5秒,让thread1先执行

        thread2.start();
        Thread.sleep(2000); // 两秒后,中断thread2

        thread2.interrupt();
    }

    public void test(Thread thread) throws InterruptedException {
        System.out.println(Thread.currentThread().getName() + ", 想获取锁");
        lock.lock();   //注意,如果需要正确中断等待锁的线程,必须将获取锁放在外面,然后将InterruptedException抛出
        try {
            System.out.println(thread.getName() + "得到了锁");
            Thread.sleep(10000); // 抢到锁,10秒不释放
        } finally {
            System.out.println(Thread.currentThread().getName() + "执行finally");
            lock.unlock();
            System.out.println(thread.getName() + "释放了锁");
        }
    }
}

结果:

Thread-0, 想获取锁
Thread-0得到了锁
Thread-1, 想获取锁
Thread-0执行finally
Thread-0释放了锁
Thread-1得到了锁
Thread-1执行finally
Thread-1释放了锁
java.lang.InterruptedException: sleep interrupted
	at java.lang.Thread.sleep(Native Method)
	at icu.shaoyayu.multithreading.chapter4.ReentrantLockDemo1.test(ReentrantLockDemo1.java:46)
	at icu.shaoyayu.multithreading.chapter4.ReentrantLockDemo1$1.run(ReentrantLockDemo1.java:24)
	at java.lang.Thread.run(Thread.java:748)

对代码进行修改

// 可响应中断
public class LockInterruptiblyDemo1 {
    private Lock lock = new ReentrantLock();

    public static void main(String[] args) throws InterruptedException {
        LockInterruptiblyDemo1 demo1 = new LockInterruptiblyDemo1();
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                try {
                    demo1.test(Thread.currentThread());
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        };
        Thread thread1 = new Thread(runnable);
        Thread thread2 = new Thread(runnable);
        thread1.start();
        Thread.sleep(500); // 等待0.5秒,让thread1先执行

        thread2.start();
        Thread.sleep(2000); // 两秒后,中断thread2

        thread2.interrupt();
    }

    public void test(Thread thread) throws InterruptedException {
        System.out.println(Thread.currentThread().getName() + ", 想获取锁");
        lock.lockInterruptibly();   //注意,如果需要正确中断等待锁的线程,必须将获取锁放在外面,然后将InterruptedException抛出
        try {
            System.out.println(thread.getName() + "得到了锁");
            Thread.sleep(10000); // 抢到锁,10秒不释放
        } finally {
            System.out.println(Thread.currentThread().getName() + "执行finally");
            lock.unlock();
            System.out.println(thread.getName() + "释放了锁");
        }
    }
}

运行结果:

Thread-0, 想获取锁
Thread-0得到了锁
Thread-1, 想获取锁
java.lang.InterruptedException
	at java.util.concurrent.locks.AbstractQueuedSynchronizer.doAcquireInterruptibly(AbstractQueuedSynchronizer.java:898)
	at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireInterruptibly(AbstractQueuedSynchronizer.java:1222)
	at java.util.concurrent.locks.ReentrantLock.lockInterruptibly(ReentrantLock.java:335)
	at icu.shaoyayu.multithreading.chapter4.LockInterruptiblyDemo1.test(LockInterruptiblyDemo1.java:42)
	at icu.shaoyayu.multithreading.chapter4.LockInterruptiblyDemo1$1.run(LockInterruptiblyDemo1.java:23)
	at java.lang.Thread.run(Thread.java:748)
Thread-0执行finally
Thread-0释放了锁

ReadWriteLock

维护一对关联锁,一个用于只读操作,一个用于写入;读锁可以由多个读线程同时持有,写锁是排他的。

适合读取线程比写入线程多的场景,改进互斥锁的性能,示例场景:缓存组件、集合的并发线程安全性改造。

锁降级指的是写锁降级成为读锁。把持住当前拥有的写锁的同时,再获取到读锁,随后释放写锁的过程。

写锁是线程独占,读锁是共享,所以写->读是升级。(读->写,是不能实现的)

示例3

// 将hashmap改造一个并发安全的
// 比hashTable的实现,效率高,读取的适合并不会同步执行
public class MapDemo {
    private final Map<String, Object> m = new HashMap<>();
    private final ReentrantReadWriteLock rwl = new ReentrantReadWriteLock();
    private final Lock r = rwl.readLock();
    private final Lock w = rwl.writeLock();

    public Object get(String key) {
        r.lock(); // 可以同时多个线程获取这把锁
        try {
            return m.get(key);
        } finally {
            r.unlock();
        }
    }

    public Object[] allKeys() {
        r.lock();
        try {
            return m.keySet().toArray();
        } finally {
            r.unlock();
        }
    }

    public Object put(String key, Object value) {
        w.lock(); // 一个线程获取 这把锁
        try {
            return m.put(key, value);
        } finally {
            w.unlock();
        }
    }

    public void clear() {
        w.lock();
        try {
            m.clear();
        } finally {
            w.unlock();
        }
    }
}

对于Hashtable而言

Hashtable源码:

package java.util;

public class Hashtable<K,V>
    extends Dictionary<K,V>
    implements Map<K,V>, Cloneable, java.io.Serializable {
    private transient Entry<?,?>[] table;
    private transient int count;
    private int threshold;
    private float loadFactor;
    private transient int modCount = 0;
    public Hashtable(int initialCapacity, float loadFactor) {
        //
    }
    public Hashtable(int initialCapacity) {
        this(initialCapacity, 0.75f);
    }
    public Hashtable() {
        //
    }
    public Hashtable(Map<? extends K, ? extends V> t) {
        //
    }
    public synchronized int size() {
        //
    }
    public synchronized boolean isEmpty() {
        //
    }
    public synchronized Enumeration<K> keys() {
        //
    }
    public synchronized Enumeration<V> elements() {
        //
    }
    public synchronized boolean contains(Object value) {
        //
    }
    public boolean containsValue(Object value) {
        //
    }
    public synchronized boolean containsKey(Object key) {
		//
    }
    @SuppressWarnings("unchecked")
    public synchronized V get(Object key) {
        //
    }
    private static final int MAX_ARRAY_SIZE = Integer.MAX_VALUE - 8;
    @SuppressWarnings("unchecked")
    protected void rehash() {
        //
    }
    private void addEntry(int hash, K key, V value, int index) {
        //
    }
    public synchronized V put(K key, V value) {
       //
    }
    public synchronized V remove(Object key) {
        //
    }
    public synchronized void putAll(Map<? extends K, ? extends V> t) {
       //
    }
    public synchronized void clear() {
        //
    }
    public synchronized Object clone() {
       //
    }
    public synchronized String toString() {
        //
    }
    private <T> Enumeration<T> getEnumeration(int type) {
        //
    }
    private <T> Iterator<T> getIterator(int type) {
        //
    }
    private transient volatile Set<K> keySet;
    private transient volatile Set<Map.Entry<K,V>> entrySet;
    private transient volatile Collection<V> values;
    public Set<K> keySet() {
        //
    }
    private class KeySet extends AbstractSet<K> {
        //
    }
    public Set<Map.Entry<K,V>> entrySet() {
       //
    }
    private class EntrySet extends AbstractSet<Map.Entry<K,V>> {
        //
    }
    public Collection<V> values() {
        //
    }
    private class ValueCollection extends AbstractCollection<V> {
        //
    }
    public synchronized boolean equals(Object o) {
        //
    }
    public synchronized int hashCode() {
        //
    }
    @Override
    public synchronized V getOrDefault(Object key, V defaultValue) {
        //
    }
    @SuppressWarnings("unchecked")
    @Override
    public synchronized void forEach(BiConsumer<? super K, ? super V> action) {
        //
    }
    @SuppressWarnings("unchecked")
    @Override
    public synchronized void replaceAll(BiFunction<? super K, ? super V, ? extends V> function) {
        //
    }
    @Override
    public synchronized V putIfAbsent(K key, V value) {
        //
    }
    @Override
    public synchronized boolean remove(Object key, Object value) {
        //
    }
    @Override
    public synchronized boolean replace(K key, V oldValue, V newValue) {
        //
    }
    @Override
    public synchronized V replace(K key, V value) {
       //
    }
    @Override
    public synchronized V computeIfAbsent(K key, Function<? super K, ? extends V> mappingFunction) {
        //
    }
    @Override
    public synchronized V computeIfPresent(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction) {
        //
    }
    @Override
    public synchronized V compute(K key, BiFunction<? super K, ? super V, ? extends V> remappingFunction) {
        //
    }
    @Override
    public synchronized V merge(K key, V value, BiFunction<? super V, ? super V, ? extends V> remappingFunction) {
        //
    }
    private void writeObject(java.io.ObjectOutputStream s)
        //
       //
    }
    private void readObject(java.io.ObjectInputStream s)
         throws IOException, ClassNotFoundException
    {
        //
    }
    private void reconstitutionPut(Entry<?,?>[] tab, K key, V value)
        throws StreamCorruptedException
    {
        //
    }
    private static class Entry<K,V> implements Map.Entry<K,V> {
        //
    }
    private static final int KEYS = 0;
    private static final int VALUES = 1;
    private static final int ENTRIES = 2;
    private class Enumerator<T> implements Enumeration<T>, Iterator<T> {
        Entry<?,?>[] table = Hashtable.this.table;
        int index = table.length;
        Entry<?,?> entry;
        Entry<?,?> lastReturned;
        int type;
        boolean iterator;
        protected int expectedModCount = modCount;
        Enumerator(int type, boolean iterator) {
            this.type = type;
            this.iterator = iterator;
        }
        public boolean hasMoreElements() {
            //
        }
        @SuppressWarnings("unchecked")
        public T nextElement() {
           //
        }
        public boolean hasNext() {
            //
        }
        public T next() {
            //
        }

        public void remove() {
           //
            synchronized(Hashtable.this) {
                //
            }
        }
    }
}

可以看到大部分的实现的都是对方法进行上锁来解决线程的高并发

示例4

读写高并发

使用synchronized关键字

// 不用读写锁
public class ReentrantReadWriteLockDemo1 {
    public static void main(String[] args)  {
        final ReentrantReadWriteLockDemo1 readWriteLockDemo1 = new ReentrantReadWriteLockDemo1();
        // 多线程同时读/写
        new Thread(() -> {
            readWriteLockDemo1.read(Thread.currentThread());
        }).start();

        new Thread(() -> {
            readWriteLockDemo1.write(Thread.currentThread());
        }).start();

        new Thread(() -> {
            readWriteLockDemo1.read(Thread.currentThread());
        }).start();
    }

    // 不管读写,只有一个线程能用, 独享锁
    public synchronized void read(Thread thread) { // 2秒
        long start = System.currentTimeMillis();
        while(System.currentTimeMillis() - start <= 1) {
            System.out.println(thread.getName()+"正在进行“读”操作");
        }
        System.out.println(thread.getName()+"“读”操作完毕");
    }

    /** 写 */
    public synchronized void write(Thread thread) {
        long start = System.currentTimeMillis();
        while(System.currentTimeMillis() - start <= 1) {
            System.out.println(thread.getName()+"正在进行“写”操作");
        }
        System.out.println(thread.getName()+"“写”操作完毕");
    }
}

使用ReentrantReadWriteLock锁实现

package icu.shaoyayu.multithreading.chapter4;

import java.util.concurrent.locks.ReentrantReadWriteLock;

/**
 * @author shaoyayu
 * @E_Mail
 * @Version 1.0.0
 * @readme :
 */
// 读写锁(既保证了读数据的效率,也保证数据的一致性)
public class ReentrantReadWriteLockDemo2 {
    ReentrantReadWriteLock readWriteLock = new ReentrantReadWriteLock();

    public static void main(String[] args) {
        final ReentrantReadWriteLockDemo2 readWriteLockDemo2 = new ReentrantReadWriteLockDemo2();
        // 多线程同时读/写
        new Thread(() -> {
            readWriteLockDemo2.read(Thread.currentThread());
        }).start();

        new Thread(() -> {
            readWriteLockDemo2.read(Thread.currentThread());
        }).start();

        new Thread(() -> {
            readWriteLockDemo2.write(Thread.currentThread());
        }).start();
    }

    // 多线程读,共享锁
    public void read(Thread thread) {
        readWriteLock.readLock().lock();
        try {
            long start = System.currentTimeMillis();
            while (System.currentTimeMillis() - start <= 1) {
                System.out.println(thread.getName() + "正在进行“读”操作");
            }
            System.out.println(thread.getName() + "“读”操作完毕");
        } finally {
            readWriteLock.readLock().unlock();
        }
    }

    /**
     * 写
     */
    public void write(Thread thread) {
        readWriteLock.writeLock().lock();
        try {
            long start = System.currentTimeMillis();
            while (System.currentTimeMillis() - start <= 1) {
                System.out.println(thread.getName() + "正在进行“写”操作");
            }
            System.out.println(thread.getName() + "“写”操作完毕");
        } finally {
            readWriteLock.writeLock().unlock();
        }
    }
}

示例5

缓存示例

// 缓存示例
public class CacheDataDemo {
    // 创建一个map用于缓存
    private Map<String, Object> map = new HashMap<>();
    private static ReadWriteLock rwl = new ReentrantReadWriteLock();

    public static void main(String[] args) {
        // 1 读取缓存里面的数据
        // cache.query()
        // 2 如果换成没数据,则取数据库里面查询  database.query()
        // 3 查询完成之后,数据塞到塞到缓存里面 cache.put(data)
    }

    public Object get(String id) {  
        Object value = null;
        // 首先开启读锁,从缓存中去取
        rwl.readLock().lock();
        try {
            if (map.get(id) == null) {
                // TODO database.query();  全部查询数据库 ,缓存雪崩
                // 必须释放读锁
                rwl.readLock().unlock();
                // 如果缓存中没有释放读锁,上写锁。如果不加锁,所有请求全部去查询数据库,就崩溃了
                rwl.writeLock().lock(); // 所有线程在此处等待  1000  1  999 (在同步代码里面再次检查是否缓存)
                try {
                    // 双重检查,防止已经有线程改变了当前的值,从而出现重复处理的情况
                    if (map.get(id) == null) {
                        // TODO value = ...如果缓存没有,就去数据库里面读取
                    }
                    rwl.readLock().lock(); // 加读锁降级写锁,这样就不会有其他线程能够改这个值,保证了数据一致性
                } finally {
                    rwl.writeLock().unlock(); // 释放写锁@
                }
            }
        } finally {
            rwl.readLock().unlock();
        }
        return value;
    }
}

问题滞留,异常的时候,释放需要释放那些锁才方便

Condition

用于替代wait/notifyo

Object中的wait(),notify(),notifyAll()方法是和synchronized配合使用的,可以唤醒一个或者全部(单个等待集);

Condition是需要与Lock配合使用的,提供多个等待集合,更精确的控制(底层是park/unpark机制);

示例6

// condition 实现队列线程安全。
public class QueueDemo {
    final Lock lock = new ReentrantLock();
    // 指定条件的等待 - 等待有空位
    final Condition notFull = lock.newCondition();
    // 指定条件的等待 - 等待不为空
    final Condition notEmpty = lock.newCondition();

    // 定义数组存储数据
    final Object[] items = new Object[100];
    int putptr, takeptr, count;

    // 写入数据的线程,写入进来
    public void put(Object x) throws InterruptedException {
        lock.lock();
        try {
            while (count == items.length) // 数据写满了
            {
                notFull.await(); // 写入数据的线程,进入阻塞
            }
            items[putptr] = x;
            if (++putptr == items.length) {
                putptr = 0;
            }
            ++count;
            notEmpty.signal(); // 唤醒指定的读取线程
        } finally {
            lock.unlock();
        }
    }
    // 读取数据的线程,调用take
    public Object take() throws InterruptedException {
        lock.lock();
        try {
            while (count == 0) {
                notEmpty.await(); // 线程阻塞在这里,等待被唤醒
            }
            Object x = items[takeptr];
            if (++takeptr == items.length) {
                takeptr = 0;
            }
            --count;
            notFull.signal(); // 通知写入数据的线程,告诉他们取走了数据,继续写入
            return x;
        } finally {
            lock.unlock();
        }
    }
}
posted @ 2020-12-02 15:12  shaoyayu  阅读(180)  评论(0编辑  收藏  举报