【代码积累】countdown latch

import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.ReentrantLock;


public class CustomLinkedBlockingQueue<T> implements CustomBlockingQueue<T>{
	private CustomHeadNode<T> head = null;  /*至少有一个头结点,尾节点*/
	private CustomTailNode<T> tail = null;  
	private ReentrantLock putlock = new ReentrantLock();
	private ReentrantLock takelock = new ReentrantLock();
	private Condition isAvailable = takelock.newCondition(); /*If no elements are available then make take actions dormant.*/
	private Condition isFull = putlock.newCondition(); /*If it's full then make put-actions dormant.*/
	/*用一个原子变量记录List的长度*/
	private AtomicInteger count = new AtomicInteger(0);
	//private CustomAtomicInteger count = new CustomAtomicInteger(0);
	private int capacity = 10; /*默认10*/
	
	public CustomLinkedBlockingQueue() {
		super();
		init();
	}
	
	public CustomLinkedBlockingQueue(int capacity) {
		if( capacity > Integer.MAX_VALUE ) {
			this.capacity = Integer.MAX_VALUE;
		} else {
			this.capacity = capacity;
		}
		
		init();
	}
	
	private void init(){
		head = new CustomHeadNode<T>();
		tail = new CustomTailNode<T>();
		head.next = tail;
		tail.prev = head;
	}
	
	/*delete and return the head of the queue*/
	private T dequeue() {
		if( count.get() > 0 && true!=head.next.isTail) {
			T value = head.next.value;
			if( null != value ) {
				/*删除中间的节点*/
				head.next.next.prev = head;
				head.next = head.next.next;
				return value;
			}
			else {
				return null;
			}
		} else {
			return null;
		}
	}
	
	/*Put the specific element to the tail of the queue*/
	private void enqueue(T ele) {
		if( count.get() < capacity ) {
			CustomNode<T> node = new CustomNode<T>();
			node.value = ele;
			tail.prev.next = node;
			node.prev = tail.prev;
			node.next = tail;
			tail.prev = node;
		}
	}
	
	@Override
	/*Take a look at the head node of the LinkedList but do not remove it.*/
	public T peek() {
		// TODO Auto-generated method stub
		return null;
	}

	@Override
	/*Remove and return the head node of the LinkedList*/
	public T poll() {
		// TODO Auto-generated method stub
		T x = null;
		int c = -1;
//		final AtomicInteger count = this.count;
//		final ReentrantLock takelock = this.takelock;
		
		try {
			takelock.lockInterruptibly();   /*Acquire the lock unless the thread is interrupted.*/
			/*当队列为空时,线程被挂起,直到被唤醒或者中断。如果线程被中断,根据takelock.lockInterruptibly()语句,线程将释放锁。*/
			while(0 == count.get()) {
				isAvailable.await();
			}
			
			/*队列不为空,则线程返回队列头元素*/
			x = dequeue();
			c = count.getAndDecrement(); /*出队列,长度计数-1*/
			if(c > 1) {  /*count在dequeue之前>1,dequeue后应该是1,还有数据,因此需要唤醒一个take线程*/
				/*还有数据可以取,再唤醒一次take线程*/
				isAvailable.signal();
			}
			
			System.out.println("queue size = "+(c-1)+" after [POLL].value="+x);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} finally {
			takelock.unlock();
		}
		
		if( c == capacity ) {
			/*队列dequeue之前满了,此时必然有put线程被挂起。
			 * 此处需要唤醒一下put线程,否则队列满后,所有的put线程被挂起,而take线程不唤醒put线程,则所有的put线程都永远挂起了。*/
			signalPutThreads();
		}
		
		return x;
	}

	@Override
	public T take() {
		// TODO Auto-generated method stub
		return poll();
	}

	@Override
	public void put(T element) {
		// TODO Auto-generated method stub
		if( element == null ) throw new NullPointerException();
		
		int c = -1;
//		final AtomicInteger count = this.count;
//		final ReentrantLock putlock = this.putlock;
		
		try {
			putlock.lockInterruptibly();
			while(count.get() == capacity) {
				isFull.await();
			}
			
			enqueue(element); /*add the element to the tail and do increment.*/
			c = count.getAndIncrement();  /*get the current value,increase 1 as the next value,return current and set next.*/
			if( c+1 < capacity ) {  
				isFull.signal();  
			}
			
			System.out.println("queue size = "+(c+1)+" after PUT.value="+element);
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} finally {
			putlock.unlock();
		}
		
		if(c == 0) {
			/*这里的c保存的是queue执行enqueue之前的值,若为0,表示之前有get线程被挂起了,因此需要唤醒一个get线程*/
			signalTakeThreads();
		}
	}
	
	private void signalTakeThreads() {
		try {
			takelock.lockInterruptibly();
			isAvailable.signal();
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} finally {
			takelock.unlock();
		}
	}
	
	private void signalPutThreads() {
		try {
			putlock.lockInterruptibly();
			isFull.signal(); /*每次唤醒一个线程,all则唤醒所有线程,但是最终执行哪个线程,取决于调度决策机制,用户无法干预。*/
		} catch (InterruptedException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		} finally {
			putlock.unlock();
		}
	}
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值