jdk源码解析(juc)的锁-java中的锁ReetrentLock

《jdk源码解析(juc)的锁-java中的锁ReetrentLock》首发橙寂博客转发请加此提示

jdk源码解析(juc)的锁-java中的锁ReetrentLock

Lock接口

Lock接口出现之前,java是通过synchronized关键字实现的锁功能,javase5之后,并发包新增了Lock接口

Lock使用方式,和分布式锁的构造很像。Lock是基于线程的,调用lock方法代表尝试获取锁,unlock代表释放锁。

Lock lock = new ReentrantLock
lock.lock();
try{
}finally{
 lock.unlock();
}

Lock接口提供了Synchronized关键字不具备的特性

尝试非阻塞地获取锁当前线程尝试获取锁,没有其他线程获取锁,则成功
能被中断的获取锁
超时获取锁在指定的时间内获取锁

Lock接口的API

api
void lock()lock 优先考虑获取锁,待获取锁成功后,才响应中断。
void lockInterruptibly() throws InterruptedExceptionlockInterruptibly 优先考虑响应中断,而不是响应锁的普通获取或重入获取。
                                    |

| boolean tryLock() | 同Lock |
| boolean tryLock(long time,TimeUtil unit) throws InterruptedException | 在一定时间time内尝试获取锁 |
| void unlock() | 释放锁 |
| Condition newCondition | 获取等待通知组件,该组件和当前的锁绑定,当前线程只有获取了锁,才能调用该组件的wait()方法,而调用后,当前线程将会释放锁 |

队列同步器AbstractQueuedSynchronized

上面介绍的是ReentrantLock中的api,最主要的事是AbstractQueuedSynchronized在做。

锁的实现基于队列同步器完成,AbstractQueuedSynchronized(简称同步器),使用一个int成员变量state表示同步状态,通过内置的FIFO队列来完成资源获取线程的排队工作
队列同步器分两种一种是NonfairSync非公平锁默认是用非公平方式,另一种是FairSync公平锁。

NonfairSync实现

所谓非公平,讲的是获取锁的方式不公平。排队买票,按理说先来先买,但是在这里每个新来的人都能跟你抢票。下面看源码的实现。
下面代码为我调用Lock方法时执行的一个顺序。其中有些方法不属于NonfairSync


   /**
    * Sync object for non-fair locks
    */
   static final class NonfairSync extends Sync {
       private static final long serialVersionUID = 7316153563782823691L;

       /**
        * Performs lock.  Try immediate barge, backing up to normal
        * acquire on failure.
        */
       final void lock() {
            //尝试着获取锁 这里也是不公平的体现,刚进来就有机会获取锁
           if (compareAndSetState(0, 1))
               setExclusiveOwnerThread(Thread.currentThread());
           else

               acquire(1);
       }


       /**
        *
        * 调用这个方法首先会尝试获取锁,如果获取不到会把当前线程加入等待列中
        */
       public final void acquire(int var1) {
           //this.acquireQueued(this.addWaiter(AbstractQueuedSynchronizer.Node.EXCLUSIVE), var1)
           //这个方法把当前线程加入等待序列。具体代码自己去看。
            if (!this.tryAcquire(var1) && this.acquireQueued(this.addWaiter(AbstractQueuedSynchronizer.Node.EXCLUSIVE), var1)) {
                selfInterrupt();
            }


       protected final boolean tryAcquire(int acquires) {
           return nonfairTryAcquire(acquires);
       }


       /**
        * Performs non-fair tryLock.  tryAcquire is implemented in
        * subclasses, but both need nonfair try for trylock method.
        *尝试获取锁
        */
       final boolean nonfairTryAcquire(int acquires) {

           //获取当前
           final Thread current = Thread.currentThread();
           //当前锁的占用量
           int c = getState();
           if (c == 0) {
           //如果没有人用这个锁尝试获取锁
               if (compareAndSetState(0, acquires)) {
                   setExclusiveOwnerThread(current);
                   return true;
               }
           }
           else if (current == getExclusiveOwnerThread()) {
               int nextc = c + acquires;
               if (nextc < 0) // overflow
                   throw new Error("Maximum lock count exceeded");
               setState(nextc);
               return true;
           }
           return false;
       }

   }
FairSync实现

所谓非公平,讲的是获取锁的方式不公平。排队买票,先来先买
下面代码为我调用Lock方法时执行的一个顺序。其中有些方法不属于NonfairSync


static final class FairSync extends Sync {
       private static final long serialVersionUID = -3000897897090466540L;

       final void lock() {
           acquire(1);
       }

       /**
        *
        * 调用这个方法首先会尝试获取锁,如果获取不到会把当前线程加入等待列中
        */
       public final void acquire(int var1) {
           //this.acquireQueued(this.addWaiter(AbstractQueuedSynchronizer.Node.EXCLUSIVE), var1)
           //这个方法把当前线程加入等待序列。具体代码自己去看。
            if (!this.tryAcquire(var1) && this.acquireQueued(this.addWaiter(AbstractQueuedSynchronizer.Node.EXCLUSIVE), var1)) {
                selfInterrupt();
            }

       /**
        * Fair version of tryAcquire.  Don't grant access unless
        * recursive call or no waiters or is first.
        */
       protected final boolean tryAcquire(int acquires) {
           final Thread current = Thread.currentThread();
           int c = getState();
           if (c == 0) {
           //这里是跟非公平锁不一样的地方
           //hasQueuedPredecessors判断线程是不是在队列中的第一个
           //只有第一个才有资格去抢
               if (!hasQueuedPredecessors() &&
                   compareAndSetState(0, acquires)) {
                   setExclusiveOwnerThread(current);
                   return true;
               }
           }
           else if (current == getExclusiveOwnerThread()) {
               int nextc = c + acquires;
               if (nextc < 0)
                   throw new Error("Maximum lock count exceeded");
               setState(nextc);
               return true;
           }
           return false;
       }
   }
release方法同unLock

字面意思释放锁,unpark掉当前线程。


/**
 * Releases in exclusive mode.  Implemented by unblocking one or
 * more threads if {@link #tryRelease} returns true.
 * This method can be used to implement method {@link Lock#unlock}.
 *
 * @param arg the release argument.  This value is conveyed to
 *        {@link #tryRelease} but is otherwise uninterpreted and
 *        can represent anything you like.
 * @return the value returned from {@link #tryRelease}
 */
public final boolean release(int arg) {
    if (tryRelease(arg)) {
        Node h = head;
        if (h != null && h.waitStatus != 0)
            unparkSuccessor(h);
        return true;
    }
    return false;
}


/**
   * Wakes up node's successor, if one exists.
   *
   * @param node the node
   */
  private void unparkSuccessor(Node node) {
      /*
       * If status is negative (i.e., possibly needing signal) try
       * to clear in anticipation of signalling.  It is OK if this
       * fails or if status is changed by waiting thread.
       */
      int ws = node.waitStatus;
      if (ws < 0)
          compareAndSetWaitStatus(node, ws, 0);

      /*
       * Thread to unpark is held in successor, which is normally
       * just the next node.  But if cancelled or apparently null,
       * traverse backwards from tail to find the actual
       * non-cancelled successor.
       */
      Node s = node.next;
      if (s == null || s.waitStatus > 0) {
          s = null;
          //从尾节点往前找,找到合适的然后干掉
          for (Node t = tail; t != null && t != node; t = t.prev)
              if (t.waitStatus <= 0)
                  s = t;
      }
      if (s != null)
          LockSupport.unpark(s.thread);
  }


使用案例

条件锁 Condition

Condition是针对于对象的,ReentrantLock是针对于线程的。

部分方法描述

void await()当前线程进入等待状态,直到被通知或中断。
void awaitUninterruptibly()当前线程进入等待状态,对中断不敏感
long awaitNanos(long nanosTimeout)当前线程进入等待状态,直到被通知,中断或者超时,返回值表示剩余的时间,返回值如果是0或者负数,那么可以认定已经超时了
boolean awaitUntil(Date deadline)当前线程进入等待状态,直到被通知、中断或者到某个时间,如果没有到指定时间,返回true,否则到了指定时间,返回false
void signal()唤醒一个等待在condition中的线程,该线程从等待方法返回前必须获取与Condition相关联的锁
void signlAll()唤醒所有等待的condition中的线程,能够从等待方法返回的线程必须获得与condition相关联的锁
public class ConditionUseCase {
    private Lock lock = new ReentrantLock();
    private Condition condition = lock.newCondition();

    public static void main(String[] args){

    }

    public void conditionWait() throws InterruptedException {
        lock.lock();
        try {
            condition.await();
        }finally {
            lock.unlock();
        }
    }
    public void conditionSignal(){
        lock.lock();
        try {
            condition.signal();
        }finally {
            lock.unlock();
        }
    }
}

通过Cache来解释读写锁,HashMap是非线程安全的,通过读写锁实现Cache的线程安全

public class Cache {
    static Map<String,Object> map = new HashMap<String,Object>();
    static ReentrantReadWriteLock rwl = new ReentrantReadWriteLock();
    static Lock r = rwl.readLock();
    static Lock w = rwl.writeLock();

    public static final Object get(String key){
        r.lock();
        try {
            return map.get(key);
        }finally {
            r.unlock();
        }
    }

    public static final Object put(String key,Object value){
        w.lock();
        try {
            return map.put(key,value);
        }finally {
            w.unlock();
        }
    }

    public static final void clear() {
        w.lock();
        try {
            map.clear();
        }finally {
            w.unlock();
        }
    }

}
  • 有界队列BoundedQueue解释Condition
public class BoundedQueue<T> {
    private Object[] items;
    private int addIndex,removeIndex,count;
    private Lock lock = new ReentrantLock();
    private Condition notEmpty = lock.newCondition();
    private Condition notFull = lock.newCondition();

    public BoundedQueue(int size){
        items = new Object[size];
    }

    public void add(T t) throws InterruptedException {
        lock.lock();
        try {
            while(count == items.length)
                notFull.await();
            items[addIndex] = t;
            if(++addIndex == items.length)
                addIndex = 0;
            ++count;
            notEmpty.signal();
        }finally {
            lock.unlock();
        }
    }

    public T remove() throws InterruptedException {
        lock.lock();
        try {
            while(count == 0)
                notEmpty.await();
            Object x = items[removeIndex];
            if(++removeIndex == items.length)
                removeIndex = 0;
            --count;
            notFull.signal();
            return (T) x;
        }finally {
            lock.unlock();
        }

    }
}

总结

ReentrantLock的讲解就到此结束,看博客前希望自己看过一点源码。如果没看过的话那么请重点关注我提出来的那几个代码。如果对于这个ReentrantLock想要更深的理解。请关注
simviso

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: aspose-words-19.3-jdk17.jar-java 是一个 Java 库,用于处理 Word 文档。它提供了一套丰富的功能和方法,允许开发人员在 Java 应用程序读取、创建、编辑和转换 Word 文档。 使用 aspose-words-19.3-jdk17.jar-java,可以轻松地将 Word 文档导入到 Java 应用程序,并以编程方式访问和修改其的内容。这个库支持各种 Word 文档格式,如 DOC、DOCX、RTF 等。可以使用它来提取文本、插入或删除段落、添加或删除图片、设置样式和格式等。 这个库还提供了强大的转换功能,可以将 Word 文档转换为其他格式,比如 PDF、HTML、EPUB 等。同时,也可以将其他格式的文档转换为 Word 文档。 aspose-words-19.3-jdk17.jar-java 在处理 Word 文档时提供了高度的可定制性。开发人员可以根据自己的需求设置字体、颜色、段落格式、页眉页脚等等。此外,它还支持在文档插入书签、超链接、表格、图表等,以满足更复杂的需求。 总之,aspose-words-19.3-jdk17.jar-java 是一个功能强大、易于使用的 Java 库,适用于处理 Word 文档。它可为开发人员提供丰富的功能和方法,让他们能够方便地读取、创建、编辑和转换 Word 文档。无论是对于个人使用还是开发团队,这个库都是一个非常有价值的工具。 ### 回答2: aspose-words-19.3-jdk17.jar是一个Java库,用于处理Word文档。它提供了丰富的功能和API,可以读取、修改和创建Word文档。通过它,我们可以在Java应用程序实现对Word文档的自动化处理。 使用aspose-words-19.3-jdk17.jar,我们可以轻松地读取现有的Word文档,并提取其的文本、表格、图像等元素。同时,还可以编辑文档的样式、格式和布局。例如,我们可以插入、删除或合并段落,设置字体、字号、颜色等。 这个库还提供了对Word文档的导出和转换功能。我们可以将文档保存为PDF、HTML、XML或其他流行的格式。此外,我们还可以将Word文档转换为其他Office应用程序的格式,如Excel或PowerPoint。 aspose-words-19.3-jdk17.jar还支持Word模板的创建和填充。我们可以使用预定义的模板,向其添加数据,并将其保存为新的Word文档。这对于生成报告、信函、合同等标准格式的文档非常有用。 总而言之,aspose-words-19.3-jdk17.jar是一个功能强大的Java库,提供了处理Word文档的丰富功能和API。它可以帮助我们在Java应用程序实现对Word文档的读取、编辑、导出和转换。无论是处理现有文档还是生成新的文档,它都是一个非常有用的工具。 ### 回答3: Aspose.Words是一个强大的Java库,用于处理和操作Microsoft Word文档。版本19.3是指该库的19.3版本,该版本具有很多功能和改进。 Aspose.Words可以在Java应用程序创建,读取,操作和转换Word文档。使用该库,您可以轻松地访问文档的内容,如文本,图片,表格,样式和格式。您可以添加,删除和修改文档的内容,包括段落,表格,图像等。您还可以应用不同的格式和样式,如字体,颜色,段落间距等。 此外,Aspose.Words还支持文档的转换功能。您可以将Word文档转换为其他格式,如PDF,HTML,纯文本等。同样,您也可以将其他格式的文件转换为Word文档。这使得Aspose.Words成为一种非常有用的工具,适用于处理和转换各种类型的文档。 Aspose.Words 19.3版特别为JDK17设计,以确保与最新的Java开发环境兼容。这意味着您可以在使用JDK17的项目无缝集成和使用Aspose.Words库,无需担心任何兼容性问题。 总而言之,Aspose.Words 19.3-jdk17.jar是一个功能强大的Java库,它为您提供了处理和操作Microsoft Word文档的能力。它支持多种功能,如创建,读取,修改和转换Word文档。此库与JDK17兼容,可以轻松地集成到Java项目,并为您的文档处理需求提供强大的支持。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值