公平锁和非公平锁,可重入锁和非可重入锁,独享锁和共享锁

1.公平锁

先进入等待队列的线程先获得锁。

2.非公平锁

每次抢占锁的时候无论先后顺序,谁拿到就归谁所有。

优点: 非公平锁是多个线程加锁时直接尝试获取锁,获取不到才会到等待队列的队尾等待。但如果此时锁刚好可用,那么这个线程可以无需阻塞直接获取到锁,所以非公平锁有可能出现后申请锁的线程先获取锁的场景。非公平锁的优点是可以减少唤起线程的开销,整体的吞吐效率高。

缺点: 处于等待队列中的线程就算先到也可能会等很久才会获得锁

3.可重入锁

可重入锁是指,当类test调用doA方法获得锁时,doA()调用doB()不会因为doA方法获得了锁而导致doB不能获得该锁。

class test(){
	synchronized doA(){
		doB();
	}
	synchronized doB(){
		System.out.println("do B");
	}
}

ReetrantLock就是典型的可重入锁。
ReetrantLock获取锁的过程:

 /**
         * 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();
            // 第一次进入,判断状态为0就获取锁,被设置c为1
            if (c == 0) {
                if (compareAndSetState(0, acquires)) {
                    setExclusiveOwnerThread(current);
                    return true;
                }
            }
            else if (current == getExclusiveOwnerThread()) {
                int nextc = c + acquires; // 重入该锁,状态c+1
                if (nextc < 0) // overflow
                    throw new Error("Maximum lock count exceeded");
                setState(nextc);
                return true;
            }
            return false;
        }

第一次获取锁就设置状态c为1(为获取锁时c=0),以后再次获取锁状态c就c++

4.不可重入锁

当class test的doA获取到了一把不可重入锁以后,doB获取不到这把锁,容易造成死锁。

5.独享锁

独享锁也叫排他锁,是指该锁一次只能被一个线程所持有。

6.共享锁

共享锁是指该锁可被多个线程所持有。

ReentrantReadWriteLock有读锁和写锁。读锁是共享锁,写锁是独享锁

公平锁和非公平锁,可重入锁和非可重入锁,独享锁和共享锁_第1张图片
写锁获取写的权限时,

protected final boolean tryAcquire(int acquires) {
	Thread current = Thread.currentThread();
	int c = getState(); // 取到当前锁的个数
	int w = exclusiveCount(c); // 取写锁的个数w
	if (c != 0) { // 如果已经有线程持有了锁(c!=0)
    // (Note: if c != 0 and w == 0 then shared count != 0)
		if (w == 0 || current != getExclusiveOwnerThread()) // 如果写线程数(w)为0(换言之存在读锁) 或者持有锁的线程不是当前线程就返回失败
			return false;
		if (w + exclusiveCount(acquires) > MAX_COUNT)    // 如果写入锁的数量大于最大数(65535,2的16次方-1)就抛出一个Error。
      throw new Error("Maximum lock count exceeded");
		// Reentrant acquire
    setState(c + acquires);
    return true;
  }
  if (writerShouldBlock() || !compareAndSetState(c, c + acquires)) // 如果当且写线程数为0,并且当前线程需要阻塞那么就返回失败;或者如果通过CAS增加写线程数失败也返回失败。
		return false;
	setExclusiveOwnerThread(current); // 如果c=0,w=0或者c>0,w>0(重入),则设置当前线程或锁的拥有者
	return true;
}

读锁:

protected final int tryAcquireShared(int unused) {
    Thread current = Thread.currentThread();
    int c = getState();
    if (exclusiveCount(c) != 0 &&
        getExclusiveOwnerThread() != current)
        return -1;                                   // 如果其他线程已经获取了写锁,则当前线程获取读锁失败,进入等待状态
    int r = sharedCount(c);
    if (!readerShouldBlock() &&
        r < MAX_COUNT &&
        compareAndSetState(c, c + SHARED_UNIT)) {
        if (r == 0) {
            firstReader = current;
            firstReaderHoldCount = 1;
        } else if (firstReader == current) {
            firstReaderHoldCount++;
        } else {
            HoldCounter rh = cachedHoldCounter;
            if (rh == null || rh.tid != getThreadId(current))
                cachedHoldCounter = rh = readHolds.get();
            else if (rh.count == 0)
                readHolds.set(rh);
            rh.count++;
        }
        return 1;
    }
    return fullTryAcquireShared(current);
}

参考文献:
https://tech.meituan.com/2018/11/15/java-lock.html

你可能感兴趣的:(并发)