JAVA并发编程有界缓存的实现详解

作者:lqh 时间:2022-12-09 16:12:53 

JAVA并发编程有界缓存的实现

1、有界缓存的基类



package cn.xf.cp.ch14;

/**
*
*功能:有界缓存实现基类
*时间:下午2:20:00
*文件:BaseBoundedBuffer.java
*@author Administrator
*
* @param <V>
*/
public class BaseBoundedBuffer<V>
{
 private final V[] buf;
 private int tail;
 private int head;
 private int count;

public BaseBoundedBuffer(int capacity)
 {
   //初始化数组
   this.buf = (V[]) new Object[capacity];
 }

//放入一个数据,final方法无法被重写
 protected synchronized final void doPut(V v)
 {
   buf[tail] = v;
   if(++tail == buf.length)
   {
     tail = 0;
   }
   //插入一个方法,总量++
   ++count;
 }

/**
  * 取出一个数据
  * @return
  */
 protected synchronized final V doTake()
 {
   V v = buf[head];
   buf[head] = null;
   if(++head == buf.length)
   {
     head = 0;
   }
   --count;
   return v;
 }

//通过对count的判断,来确定数组是否是满的
 public synchronized final boolean isFull()
 {
   return count == buf.length;
 }

public synchronized final boolean isEmpty()
 {
   return count == 0;
 }
}

2、判定前提条件再执行操作



package cn.xf.cp.ch14;

/**
*
*功能:对插入和获取元素操作进行先行检查,然后执行操作,校验不通过不予操作
*时间:下午2:33:41
*文件:GrumpyBoundedBuffer.java
*@author Administrator
*
* @param <V>
*/
public class GrumpyBoundedBuffer<V> extends BaseBoundedBuffer<V>
{

public GrumpyBoundedBuffer(int size)
 {
   super(size);
 }

public synchronized void put(V v) throws Exception
 {
   //如果是满的队列,就无法插入新的元素
   if(this.isFull())
   {
     throw new Exception("队列超出");
   }
   this.doPut(v);
 }

//同理,队列为空的就无法取出新的元素
 public synchronized V take() throws Exception
 {
   if(this.isEmpty())
   {
     throw new Exception("队列中无元素");
   }

return this.doTake();
 }

}

3、通过轮询与休眠来实现简单的阻塞



package cn.xf.cp.ch14;

/**
*
*功能:通过轮询与休眠来实现简单的阻塞
*时间:下午2:55:54
*文件:SleepyBoundedBuffer.java
*@author Administrator
*
* @param <V>
*/
public class SleepyBoundedBuffer<V> extends BaseBoundedBuffer<V>
{
 //2s
 private static final long SLEEP_GRANULARITY = 2000;

public SleepyBoundedBuffer(int capacity)
 {
   super(capacity);
 }

//放入队列的时候
 public void put(V v) throws InterruptedException
 {
   while(true)
   {
     //这里不对循环上锁,不然这个锁就无法释放了,不对休眠上锁,休眠上锁,在休眠的时候别人也无法操作,永远都不可能有元素出去
     synchronized (this)
     {
       //如果队列不是满的,那么就放入元素
       if(!this.isFull())
       {
         this.doPut(v);
         return;
       }
     }
     //否则休眠,退出cpu占用
     Thread.sleep(SLEEP_GRANULARITY);
   }
 }

public V take() throws InterruptedException
 {
   while(true)
   {
     //这里不对循环上锁,不然这个锁就无法释放了,不对休眠上锁,休眠上锁,在休眠的时候别人也无法操作,永远都不可能有新的元素进来
     synchronized(this)
     {
       //如果数组部位空,那么就可以取出数据
       if(!this.isEmpty())
       {
         return this.doTake();
       }
       //如果队列为空,休眠几秒再试
     }
     Thread.sleep(SLEEP_GRANULARITY);
   }
 }

}

4、条件队列



package cn.xf.cp.ch14;

/**
*
*功能:使用条件队列
*时间:下午3:32:04
*文件:BoundedBuffer.java
*@author Administrator
*
* @param <V>
*/
public class BoundedBuffer<V> extends BaseBoundedBuffer<V>
{

public BoundedBuffer(int capacity)
 {
   super(capacity);
 }

/**
  * 放入数据元素
  * @param v
  * @throws InterruptedException
  */
 public synchronized void put(V v) throws InterruptedException
 {
   while(this.isFull())
   {
     //这里挂起程序,会释放锁
     this.wait();
   }
   //如果队列不为满的,那么程序被唤醒之后从新获取锁
   this.doPut(v);
   //执行结束,唤醒其他队列
   this.notifyAll();
 }

public synchronized V take() throws InterruptedException
 {
   while(this.isEmpty())
   {
     this.wait();
   }
   V v = this.doTake();
   this.notifyAll();
   return v;
 }

}

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

标签:java,有界缓存
0
投稿

猜你喜欢

  • Android自定义ProgressDialog加载图片

    2022-07-28 22:08:39
  • 轻松实现Android自定义九宫格图案解锁

    2023-09-20 00:32:52
  • Spring Boot 2.X快速整合jpa过程解析

    2021-09-22 15:27:29
  • Java的类型擦除式泛型详解

    2022-01-02 11:59:00
  • Java算法之时间复杂度和空间复杂度的概念和计算

    2023-06-11 17:47:56
  • 详解.NET 6如何实现获取当前登录用户信息

    2022-01-29 12:51:15
  • 详解Java策略模式

    2022-03-20 12:17:05
  • 深入了解Java核心类库--Math类

    2023-08-19 01:06:21
  • Java探索之Feign入门使用详解

    2023-08-18 03:38:06
  • C# Csv实现基本的读写和转换DataTable

    2023-11-20 10:55:22
  • Java Socket编程实例(四)- NIO TCP实践

    2021-10-07 13:44:35
  • java线程池:获取运行线程数并控制线程启动速度的方法

    2022-06-30 23:22:57
  • C#使用委托(delegate)实现在两个form之间传递数据的方法

    2022-07-04 21:07:54
  • Android EditText限制输入字符类型的方法总结

    2023-10-11 15:48:19
  • Android 实现签到足迹功能

    2023-06-21 15:16:24
  • C#中定时任务被阻塞问题的解决方法

    2023-10-27 00:56:02
  • 人工智能开发语言排行榜: 不死Java, 不朽C/C++, 新贵Python【推荐】

    2023-03-12 16:06:54
  • Android 获得View宽高的几种方式总结

    2021-11-12 05:06:49
  • flutter TextField换行自适应的实现

    2023-06-21 01:21:39
  • 解决mybatis分页插件PageHelper导致自定义拦截器失效

    2023-10-13 07:24:29
  • asp之家 软件编程 m.aspxhome.com