JAVA并发之AQS

https://www.cnblogs.com/waterystone/p/4920797.html

/**

  • Provides a framework for implementing blocking locks and related
  • synchronizers (semaphores, events, etc) that rely on
  • first-in-first-out (FIFO) wait queues. This class is designed to
  • be a useful basis for most kinds of synchronizers that rely on a
  • single atomic {@code int} value to represent state. Subclasses
  • must define the protected methods that change this state, and which
  • define what that state means in terms of this object being acquired
  • or released. Given these, the other methods in this class carry
  • out all queuing and blocking mechanics. Subclasses can maintain
  • other state fields, but only the atomically updated {@code int}
  • value manipulated using methods {@link #getState}, {@link
  • setState} and {@link #compareAndSetState} is tracked with respect

  • to synchronization.
  • <p>Subclasses should be defined as non-public internal helper
  • classes that are used to implement the synchronization properties
  • of their enclosing class. Class
  • {@code AbstractQueuedSynchronizer} does not implement any
  • synchronization interface. Instead it defines methods such as
  • {@link #acquireInterruptibly} that can be invoked as
  • appropriate by concrete locks and related synchronizers to
  • implement their public methods.
  • <p>This class supports either or both a default <em>exclusive</em>
  • mode and a <em>shared</em> mode. When acquired in exclusive mode,
  • attempted acquires by other threads cannot succeed. Shared mode
  • acquires by multiple threads may (but need not) succeed. This class
  • does not "understand" these differences except in the
  • mechanical sense that when a shared mode acquire succeeds, the next
  • waiting thread (if one exists) must also determine whether it can
  • acquire as well. Threads waiting in the different modes share the
  • same FIFO queue. Usually, implementation subclasses support only
  • one of these modes, but both can come into play for example in a
  • {@link ReadWriteLock}. Subclasses that support only exclusive or
  • only shared modes need not define the methods supporting the unused mode.
  • <p>This class defines a nested {@link ConditionObject} class that
  • can be used as a {@link Condition} implementation by subclasses
  • supporting exclusive mode for which method {@link
  • isHeldExclusively} reports whether synchronization is exclusively

  • held with respect to the current thread, method {@link #release}
  • invoked with the current {@link #getState} value fully releases
  • this object, and {@link #acquire}, given this saved state value,
  • eventually restores this object to its previous acquired state. No
  • {@code AbstractQueuedSynchronizer} method otherwise creates such a
  • condition, so if this constraint cannot be met, do not use it. The
  • behavior of {@link ConditionObject} depends of course on the
  • semantics of its synchronizer implementation.
  • <p>This class provides inspection, instrumentation, and monitoring
  • methods for the internal queue, as well as similar methods for
  • condition objects. These can be exported as desired into classes
  • using an {@code AbstractQueuedSynchronizer} for their
  • synchronization mechanics.
  • <p>Serialization of this class stores only the underlying atomic
  • integer maintaining state, so deserialized objects have empty
  • thread queues. Typical subclasses requiring serializability will
  • define a {@code readObject} method that restores this to a known
  • initial state upon deserialization.
  • <h3>Usage</h3>
  • <p>To use this class as the basis of a synchronizer, redefine the
  • following methods, as applicable, by inspecting and/or modifying
  • the synchronization state using {@link #getState}, {@link
  • setState} and/or {@link #compareAndSetState}:

  • <ul>
  • <li> {@link #tryAcquire}
  • <li> {@link #tryRelease}
  • <li> {@link #tryAcquireShared}
  • <li> {@link #tryReleaseShared}
  • <li> {@link #isHeldExclusively}
  • </ul>
  • Each of these methods by default throws {@link
  • UnsupportedOperationException}. Implementations of these methods
  • must be internally thread-safe, and should in general be short and
  • not block. Defining these methods is the <em>only</em> supported
  • means of using this class. All other methods are declared
  • {@code final} because they cannot be independently varied.
  • <p>You may also find the inherited methods from {@link
  • AbstractOwnableSynchronizer} useful to keep track of the thread
  • owning an exclusive synchronizer. You are encouraged to use them
  • -- this enables monitoring and diagnostic tools to assist users in
  • determining which threads hold locks.
  • <p>Even though this class is based on an internal FIFO queue, it
  • does not automatically enforce FIFO acquisition policies. The core
  • of exclusive synchronization takes the form:
  • <pre>
  • Acquire:
  • while (!tryAcquire(arg)) {
    
  •    <em>enqueue thread if it is not already queued</em>;
    
  •    <em>possibly block current thread</em>;
    
  • }
    
  • Release:
  • if (tryRelease(arg))
    
  •    <em>unblock the first queued thread</em>;
    
  • </pre>
  • (Shared mode is similar but may involve cascading signals.)
  • <p id="barging">Because checks in acquire are invoked before
  • enqueuing, a newly acquiring thread may <em>barge</em> ahead of
  • others that are blocked and queued. However, you can, if desired,
  • define {@code tryAcquire} and/or {@code tryAcquireShared} to
  • disable barging by internally invoking one or more of the inspection
  • methods, thereby providing a <em>fair</em> FIFO acquisition order.
  • In particular, most fair synchronizers can define {@code tryAcquire}
  • to return {@code false} if {@link #hasQueuedPredecessors} (a method
  • specifically designed to be used by fair synchronizers) returns
  • {@code true}. Other variations are possible.
  • <p>Throughput and scalability are generally highest for the
  • default barging (also known as <em>greedy</em>,
  • <em>renouncement</em>, and <em>convoy-avoidance</em>) strategy.
  • While this is not guaranteed to be fair or starvation-free, earlier
  • queued threads are allowed to recontend before later queued
  • threads, and each recontention has an unbiased chance to succeed
  • against incoming threads. Also, while acquires do not
  • "spin" in the usual sense, they may perform multiple
  • invocations of {@code tryAcquire} interspersed with other
  • computations before blocking. This gives most of the benefits of
  • spins when exclusive synchronization is only briefly held, without
  • most of the liabilities when it isn't. If so desired, you can
  • augment this by preceding calls to acquire methods with
  • "fast-path" checks, possibly prechecking {@link #hasContended}
  • and/or {@link #hasQueuedThreads} to only do so if the synchronizer
  • is likely not to be contended.
  • <p>This class provides an efficient and scalable basis for
  • synchronization in part by specializing its range of use to
  • synchronizers that can rely on {@code int} state, acquire, and
  • release parameters, and an internal FIFO wait queue. When this does
  • not suffice, you can build synchronizers from a lower level using
  • {@link java.util.concurrent.atomic atomic} classes, your own custom
  • {@link java.util.Queue} classes, and {@link LockSupport} blocking
  • support.
  • <h3>Usage Examples</h3>
  • <p>Here is a non-reentrant mutual exclusion lock class that uses
  • the value zero to represent the unlocked state, and one to
  • represent the locked state. While a non-reentrant lock
  • does not strictly require recording of the current owner
  • thread, this class does so anyway to make usage easier to monitor.
  • It also supports conditions and exposes
  • one of the instrumentation methods:
  • <pre> {@code
  • class Mutex implements Lock, java.io.Serializable {
  • // Our internal helper class
  • private static class Sync extends AbstractQueuedSynchronizer {
  • // Reports whether in locked state
    
  • protected boolean isHeldExclusively() {
    
  •   return getState() == 1;
    
  • }
    
  • // Acquires the lock if state is zero
    
  • public boolean tryAcquire(int acquires) {
    
  •   assert acquires == 1; // Otherwise unused
    
  •   if (compareAndSetState(0, 1)) {
    
  •     setExclusiveOwnerThread(Thread.currentThread());
    
  •     return true;
    
  •   }
    
  •   return false;
    
  • }
    
  • // Releases the lock by setting state to zero
    
  • protected boolean tryRelease(int releases) {
    
  •   assert releases == 1; // Otherwise unused
    
  •   if (getState() == 0) throw new IllegalMonitorStateException();
    
  •   setExclusiveOwnerThread(null);
    
  •   setState(0);
    
  •   return true;
    
  • }
    
  • // Provides a Condition
    
  • Condition newCondition() { return new ConditionObject(); }
    
  • // Deserializes properly
    
  • private void readObject(ObjectInputStream s)
    
  •     throws IOException, ClassNotFoundException {
    
  •   s.defaultReadObject();
    
  •   setState(0); // reset to unlocked state
    
  • }
    
  • }
  • // The sync object does all the hard work. We just forward to it.
  • private final Sync sync = new Sync();
  • public void lock() { sync.acquire(1); }
  • public boolean tryLock() { return sync.tryAcquire(1); }
  • public void unlock() { sync.release(1); }
  • public Condition newCondition() { return sync.newCondition(); }
  • public boolean isLocked() { return sync.isHeldExclusively(); }
  • public boolean hasQueuedThreads() { return sync.hasQueuedThreads(); }
  • public void lockInterruptibly() throws InterruptedException {
  • sync.acquireInterruptibly(1);
    
  • }
  • public boolean tryLock(long timeout, TimeUnit unit)
  •   throws InterruptedException {
    
  • return sync.tryAcquireNanos(1, unit.toNanos(timeout));
    
  • }
  • }}</pre>
  • <p>Here is a latch class that is like a
  • {@link java.util.concurrent.CountDownLatch CountDownLatch}
  • except that it only requires a single {@code signal} to
  • fire. Because a latch is non-exclusive, it uses the {@code shared}
  • acquire and release methods.
  • <pre> {@code
  • class BooleanLatch {
  • private static class Sync extends AbstractQueuedSynchronizer {
  • boolean isSignalled() { return getState() != 0; }
    
  • protected int tryAcquireShared(int ignore) {
    
  •   return isSignalled() ? 1 : -1;
    
  • }
    
  • protected boolean tryReleaseShared(int ignore) {
    
  •   setState(1);
    
  •   return true;
    
  • }
    
  • }
  • private final Sync sync = new Sync();
  • public boolean isSignalled() { return sync.isSignalled(); }
  • public void signal() { sync.releaseShared(1); }
  • public void await() throws InterruptedException {
  • sync.acquireSharedInterruptibly(1);
    
  • }
  • }}</pre>
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,588评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,456评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,146评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,387评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,481评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,510评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,522评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,296评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,745评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,039评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,202评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,901评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,538评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,165评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,415评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,081评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,085评论 2 352

推荐阅读更多精彩内容