Java AQS

AQS(抽象队列同步器) 代表的是AbstractQueuedSynchronizer类的缩写,该类来自JDK1.5,是JDK提供用于实现依赖于先进先出(FIFO)等待队列的阻塞锁和相关同步器的框架,其类的定义如下:

public abstract class AbstractQueuedSynchronizer
    extends AbstractOwnableSynchronizer
    implements java.io.Serializable

该类存在几个重要的属性和内部类:

private volatile int state; // 加锁状态
// 下面三个关于state的操作,全部是原子操作
protected final int getState() {
    return state;
}
protected final void setState(int newState) {
    state = newState;
}
protected final boolean compareAndSetState(int expect, int update) {
    return unsafe.compareAndSwapInt(this, stateOffset, expect, update);
}

private transient volatile Node head; // CLH双向队列的头部
private transient volatile Node tail; // CLH双向队列的尾部

private transient Thread exclusiveOwnerThread; // 当前线程

// 同步条件
public class ConditionObject implements Condition, java.io.Serializable {...}
// CLH 双向队列
static final class Node {...}

AQS 重要方法分析(其内部使用了一种设计模式,模板方法模式):

AQS定义两种资源共享方式:Exclusive(独占,只有一个线程能执行,如ReentrantLock)和Share(共享,多个线程可同时执行,如Semaphore/CountDownLatch)。

// 独占锁获取锁
final void acquire(int arg) {
	if (!tryAcquire(arg) &&
		acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
		selfInterrupt();
}
// 该方法需要被实现
protected boolean tryAcquire(int arg) {
	throw new UnsupportedOperationException();
}
// 独占锁释放锁
public final boolean release(int arg) {
	if (tryRelease(arg)) {
		Node h = head;
		if (h != null && h.waitStatus != 0)
			unparkSuccessor(h);
		return true;
	}
	return false;
}
// 该方法需要被实现
protected boolean tryRelease(int arg) {
	throw new UnsupportedOperationException();
}

// 共享锁获取锁
public final void acquireShared(int arg) {
	if (tryAcquireShared(arg) < 0)
		doAcquireShared(arg);
}
// 该方法需要被实现
protected int tryAcquireShared(int arg) {
	throw new UnsupportedOperationException();
}
// 共享锁释放锁
public final boolean releaseShared(int arg) {
	if (tryReleaseShared(arg)) {
		doReleaseShared();
		return true;
	}
	return false;
}
// 该方法需要被实现
protected boolean tryReleaseShared(int arg) {
	throw new UnsupportedOperationException();
}

// 判断该线程是否占领资源
// 该方法需要被实现
protected boolean isHeldExclusively() {
	throw new UnsupportedOperationException();
}

AQS 简单的自定义锁实现:

import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.AbstractQueuedSynchronizer;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;

public class MyLock implements Lock {
    private Sync sync = new Sync();

    @Override
    public void lock() {
        sync.acquire(1);
    }

    @Override
    public void lockInterruptibly() throws InterruptedException {

    }

    @Override
    public boolean tryLock() {
        return false;
    }

    @Override
    public boolean tryLock(long time, TimeUnit unit) throws InterruptedException {
        return false;
    }

    @Override
    public void unlock() {
        sync.release(1);
    }

    @Override
    public Condition newCondition() {
        return null;
    }
    
    // 自定义同步器在实现时只需要实现共享资源state的获取与释放方式即可,至于具体线程等待队列的维护,AQS已经在顶层实现好了
    protected class Sync extends AbstractQueuedSynchronizer {
        /**
         * 独占方式,尝试获取资源(共享方式tryAcquireShared)
         * @param arg
         * @return
         */
        @Override
        protected boolean tryAcquire(int arg) {
            assert arg == 1;
            // compareAndSetState 原子操作(CAS)
            if (compareAndSetState(0, 1)) {
                setExclusiveOwnerThread(Thread.currentThread());
                return true;
            }
            return false;
        }

        /**
         * 独占方式,尝试释放资源(共享方式tryReleaseShared)
         * @param arg
         * @return
         */
        @Override
        protected boolean tryRelease(int arg) {
            assert arg == 1;
            if (!isHeldExclusively()) throw new IllegalMonitorStateException();
            setExclusiveOwnerThread(null);
            setState(0);
            return true;
        }

        /**
         * 该线程是否正在独占资源
         * @return
         */
        @Override
        protected boolean isHeldExclusively() {
            return getExclusiveOwnerThread() == Thread.currentThread();
        }
    }
}

总结:

AQS是JDK提供用于实现依赖于先进先出(FIFO)等待队列的阻塞锁和相关同步器的框架。例如ReentrantLock、Semaphore、CountDownLatch、ReentrantReadWriteLock、SynchronousQueue、FutureTask等等都是基于AQS实现的。当然我们也可以利用AQS很方便的实现符合自己要求的同步器。
AQS核心思想是,如果被请求的共享资源空闲,则将当前请求资源的线程设置为有效的工作线程,并且将共享资源设置为锁定状态。如果被请求的共享资源被占用,那么就需要一套线程阻塞等待以及被唤醒时锁分配的机制,这个机制AQS是用CLH队列实现的,即将暂时获取不到锁的线程加入到等待队列中。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值