JAVA循环队列

关于自定义循环队列的实现原理和要点可以参见之前的博文系列:循环队列及C语言实现。这里主要对JAVA下的具体实现方式与原理进行说明。

一、JAVA 中已经自带了 Queue、DQueue、ArrayList、LinkedList 等常用的数据结构,为什么还要单独实现循环队列?

之所以使用自定义循环队列,出发点还是基于我们在实际应用中对于数据处理各种各样的需求。使用自定义数据结构的好处就在于可以更加灵活的处理各种数据,增加自己需要的接口。然而弊端就是你的 code 可能会引入各种未知的 Bug。所以,在满足我们使用前提的场景下,使用上述已有数据结构等是比较推荐的,如果满足不了实际项目需求,再通过自定义的方式等实现是我们理想的一种选择。

二、什么场景需要使用自定义循环队列?

对于数据的处理要求比较灵活,比如:我们需要开发一个安卓服务器程序,需要不断处理每个客户端的请求以及完成与客户端的交互。发送来的数据需要及时处理,但同时数据以自定义格式进行传输:包头+长度+数据+包尾+校验。如上格式,服务器端需要不断将接收的数据进行缓存与解析,如果未满一帧那么需要缓存到下次数据接收过来再进行解析。这时,我们需要批量从队列读取数据以及如果一帧数据不完全,将读取的数据复原到队列中(更改队列当前读位置)的功能。此时,就可以考虑自己实现队列满足这些特殊的需求。

三、循环队列的特点与要素

1、先进先出(FIFO);

2、队列首尾元素位置;

3、常用队列操作:初始化、销毁、遍历、读写等;

四、源码实现

为便于使用,这里将该循环队列以类的方式实现:

/** Copyright (c) 2017, SoldierJazz. All rights reserved.* Use is subject to license terms.**/package com.exmple.java.text;/*** DataQueue 类实现为FIFO循环队列** <p> 使用前需要根据实际需求为队列分配合理的队列空间大小** 创建一个4K空间的队列如下所示:** DataQueue mdataqueue= new DataQueue(4096);** <p> 更多使用信息可以参考引用该类的例程,有关问题,可发送到* SoldierJazz@163.com 寻求支持。** @author      SoldierJazz* @version	 1.0.0*/public class DataQueue {Queue q = null;public class Queue {byte[] data = null;int read;int write;int size;int space;}Object mSemaphore = new Object();DataQueue(int size) {q = new Queue();Queue_Init(q, size);}/*** 返回当前队列可用数据量** @param    q    目标队列**/int Avail(Queue q) {return q.size - q.space;}/*** 初始化队列** @param    q    目标队列* * @param    size    队列分配内存大小**/void Queue_Init(Queue q, int size) {synchronized (mSemaphore) {q.data = new byte[size];q.read = 0;q.write = 0;q.size = size;q.space = size;}}/*** 销毁队列** @param    q    目标队列**/void Queue_Destroy(Queue q) {synchronized (mSemaphore) {q.read = q.write = 0;q.space = q.size;}}/*** 判断当前队列是否为空** @param    q    目标队列** @return    true表示队列为空<br>false表示队列不为空**/boolean Queue_Empty(Queue q) {return (q.space == q.size);}/*** 判断当前队列是否已满** @param    q    目标队列** @return    true表示队列已满<br>false表示队列未满**/boolean Queue_Full(Queue q) {return (q.space == 0);}/*** 写一个byte到目标队列** @param    q    目标队列* * @param    val    写入的byte值** @return    true表示写入成功<br>false表示写入失败**/boolean AddQueue(Queue q, byte val) {if (!Queue_Full(q)) {q.data[q.write] = val;q.write = (q.write + 1) % q.size;q.space--;return true;} return false;}/*** 从队列中读取一个字节** @param    q    目标队列* * @param    data    读取的字节** @return    true表示读取成功<br>false表示读取失败**/boolean DelQueue(Queue q, Byte data) {if (!Queue_Empty(q)) {data = q.data[q.read];q.read = (q.read + 1) % q.size;q.space++;return true;}return false;}/*** 批量写入长度为len的字节到队列** @param    q    目标队列* * @param    data    写入的byte数组* * @param    len    写入的数组长度** @return    成功写入的字节数量**/int WriteQueue(Queue q, byte[] data, int len){int ret = 0;int rest = q.size - q.write;synchronized (mSemaphore) {if (!Queue_Full(q)) {if (q.space >= len) {ret = len;if (rest >= len) {System.arraycopy(data, 0, q.data, q.write, len);q.write = (q.write + len) % q.size;q.space -= len;} else {System.arraycopy(data, 0, q.data, q.write, rest);q.write = 0;System.arraycopy(data, rest, q.data, 0, len - rest);q.write = len -rest;q.space -= len;}} else {ret = q.space;if (rest >= q.space) {System.arraycopy(data, 0, q.data, q.write, q.space);q.write = (q.write + q.space) % q.size;q.space = 0;} else {System.arraycopy(data, 0, q.data, q.write, rest);q.write = 0;System.arraycopy(data, rest, q.data, 0, q.space - rest);q.write = q.space -rest;q.space = 0;}}   }return ret;}}/*** 从队列中恢复长度len个字节的数据** @param    q    目标队列* * @param    len    要恢复的长度** @return    成功恢复的字节数**/int RecoverReadQueue(Queue q, int len) {int ret = 0;int rest = q.read;synchronized (mSemaphore) {if (q.space >= len)ret = len;elseret = q.space;if (rest >= ret) {q.read -= ret;} else {q.read = q.size - (ret - rest);}q.space -= ret;return ret;}}/*** 从队列中读取len个字节数据到data数组中** @param    q    目标队列* * @param    data    用于存放数据的目标数组* * @param    start    拷贝至目标数组的起始位置* * @param    len    读取的长度* * @return    成功读取的字节数**/int ReadQueue(Queue q, byte[] data, int start, int len) {int rest = q.size - q.read;int ret = 0;synchronized (mSemaphore) {if (!Queue_Empty(q)) {if (Avail(q) >= len) {ret = len;if (rest >= len) {System.arraycopy(q.data, q.read, data, start, len);q.read = (q.read + len) % q.size;q.space += len;} else {System.arraycopy(q.data, q.read, data, start, rest);q.read = 0;System.arraycopy(q.data, 0, data, start + rest, len - rest);q.read = len -rest;q.space += len;}return len;} else {ret = Avail(q);if (rest >= Avail(q)) {System.arraycopy(q.data, q.read, data, start, Avail(q));q.read = (q.read + Avail(q)) % q.size;q.space = q.size;} else {System.arraycopy(q.data, q.read, data, start, rest);q.read = 0;System.arraycopy(q.data, 0, data, start + rest, Avail(q) - rest);q.read = Avail(q) -rest;q.space = q.size;}}} return ret;}}
}
以上内容为使用该类及相关方法的定义,比较简单,看注解即可。下面针对该类做一个使用与测试程序:

	public void TestDataQueue() {DataQueue dataq = new DataQueue(100);byte[] a1 = {1, 2, 3, 4, 5, 6};byte[] a2 = {7, 8, 9, 10};byte[] b = new byte[10];int nread = 0;dataq.WriteQueue(dataq.q, a1, a1.length);nread = dataq.ReadQueue(dataq.q, b, 0, 3);System.out.println("length of queue: " + dataq.Avail(dataq.q));for (int i = 0; i < nread; i++) {System.out.printf("byte[%d]: %d\n", i, b[i]);}dataq.WriteQueue(dataq.q, a2, a2.length);System.out.println("length of queue: " + dataq.Avail(dataq.q));nread = dataq.ReadQueue(dataq.q, b, 0, dataq.Avail(dataq.q));System.out.println("length of queue: " + dataq.Avail(dataq.q));for (int i = 0; i < nread; i++) {System.out.printf("byte[%d]: %d\n", i, b[i]);}}public static void main(String args[]) {test t = new test();t.TestDataQueue();}
运行结果如下所示:



有疑问或者问题就给我邮件或者评论吧,觉得有用就点赞吧~:-D



本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/403048.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

VMware 虚拟机占用磁盘空间

使用VMware创建的虚拟机尽管已经设定分配的磁盘大小&#xff0c;但仍然会发现虚拟机占用的磁盘空间会越来越大&#xff0c;而直观体现就是虚拟机系统文件 vmdk 不断增大。因此下面介绍一个简单的方法&#xff0c;使用 VMware 自带的工具对 vmdk 文件进行压缩以节省磁盘空间。拿…

frameworks/av/media/CedarX-Projects/CedarAndroidLib/LIB_KK44_/Android.mk: No such file or directory

在安卓系统编译过程中如果遇到上述或者与之类似的错误&#xff0c;可以采取相同的处理方法进行解决。直接进入到 CedarAndroidLib 目录下&#xff0c;也即此例中的 frameworks/av/media/CedarX-Projects/CedarAndroidLib。看一下当前文件&#xff1a; 注意第9行为包含标题中报错…

Windows与Linux下tftp服务的使用

tftp 协议是基于 udp 的&#xff0c;轻量小巧&#xff0c;用在局域网和嵌入式上很顺手。大部分帖子把在 linux 上配置的过程描述的过于复杂&#xff0c;其实只是个工具而已。研究协议抓下包对比协议内容也可以满足需求了&#xff0c;下面进入正文。分别讲下在 linux 以及 windo…

饥荒Mod 开发(二二):显示物品信息

饥荒Mod 开发(二一)&#xff1a;超大便携背包&#xff0c;超大物品栏&#xff0c;永久保鲜 饥荒中的物品没有详细信息&#xff0c;基本上只有一个名字&#xff0c;所以很多物品的功能都不知道&#xff0c;比如浆果吃了也不知道恢复什么&#xff0c; 采集的胡萝卜也不知道什么功…

安卓BLE开发教程(一) BLE基础

我试图以一种简单的方式去把重要的事情讲清楚。目的是希望BLE协议栈和基础概念简单化&#xff0c;让自己及类似的安卓开发者可以在较短的时间内把握住BLE的核心及使用方法。BLE本身很复杂&#xff0c;但对于安卓开发而言只要抓住一些核心点&#xff0c;便已足够。如果你想全面了…

Linux驱动如何在不同版本上快速迭代升级

As well known&#xff0c;Linux内核版本更新很快&#xff0c;有些内核版本的迭代升级可能会导致在使用的驱动版本存在编译失败或使用的兼容性问题&#xff0c;如何快速定位到内核版本间变更的地方&#xff0c;并处理掉该问题&#xff0c;列一下我常用的解决方法。&#xff08;…

苹果MacOS系统上安装第三方驱动失败/无效

近期不少用户在苹果系统上安装一些第三方驱动时反馈没有作用&#xff0c;但是驱动安装提示是完成的&#xff0c;并拷贝到了系统的驱动路径下&#xff1b;造成该问题的原因可参见如下苹果官方说明&#xff1a; User-Approved Kernel Extension Loading 引用下第一段官网说明 …

OpenWrt 之 MT7628 移植第三方SPI驱动

1、在OpenWrt系统上移植SPI驱动前&#xff0c;首先要确保SPI相关引脚未被复用为其他功能&#xff0c;比如GPIO&#xff1b;以下操作已假定该条件成立&#xff0c;否则请修改相关dts和c文件中复用配置&#xff1b; 2、打开dts配置文件进行修改&#xff0c;这里我是用的硬件为WR…

CH9102 USB转串口应用体验

近期使用CH9102 USB转串口芯片成功用在原有使用CP2102的产品板上&#xff0c;整个替换和验证过程还是很顺利的&#xff0c;顺带写个blog做个记录。 原项目上使用CP2102搭载ESP32实现Arduino物联网应用&#xff0c;采用USB转串口芯片实现串口下载&#xff0c;代码的Debug调试&am…

CH9101 USB转串口替换FT232R和FT230XQ

学生党一枚&#xff0c;前段时间跟着导师做的项目因为上面用到USB转串口芯片FT232R迟迟买不到&#xff0c;所以打算更换成国产USB转串口芯片CH340&#xff0c;对CH340的认识也很早了&#xff0c;很多年前开始直到现在各种开发板上基本都会标配一颗CH340&#xff0c;像某宝上的S…

FT230X芯片的国产化替代

之前有些项目用到FT230XQ芯片&#xff0c;无奈不好买&#xff08;价格高&#xff09;&#xff0c;想找些替代的型号。原先使用国产CH340芯片比较多&#xff0c;顺带去官网找下有没有其他小封装的芯片型号。导航比较方便&#xff0c;从官网的产品中心&#xff0c;选择“USB”分类…

Java Code之多态

Java代码 package com.iteye.badpie.javacode.duotai; /** * 人民警察 */public interface IPolice { /** * 抓小偷 */public void catchThief(); }package com.iteye.badpie.javacode.duotai;/*** 人民警察*/ public interface IPolice {/*** 抓小偷*/public void catch…

最小生成树之prim

prim是设置一个初始结点&#xff0c;寻找其周围最小的边权值&#xff0c;并将该结点作为初始结点&#xff0c;继续寻找现在结点周围的边权值的最小值&#xff0c;但要注意如果这次寻找的某个边权值没有上次的小的话仍然保留上一次的边权值&#xff0c;即lowcast的值将会不变。 …

element-ui自定义表头;el-table自定义表头;render-header自定义表头

自定义表头有两种方式&#xff1a;一种是使用render-header 一种是通过设置 Scoped slot 来自定义表头 一、render-header方式 场景&#xff1a;给表头设置自定义按钮&#xff0c;点击时候 批量下载或做其他事件 给当前的那列设置 :render-header <el-table-column align&…

vue项目转rem;H5配置rem;px转rem

H5可以配合vant组件库书写项目&#xff0c;和使用rem后vant组件样式变小了解决办法。&#xff08;引入方式&#xff09; 以下是配置rem步骤&#xff1a; 1.安装 flexible和 postcss-px2rem&#xff08;命令行安装&#xff09; lib-flexible 会自动在为你添加 meta name“viewp…

git代码回滚到以前某一版本

1.使用 git log 查看之前提交的版本&#xff0c;每一版对应的hash值&#xff0c;默认展示几条&#xff0c;如果想查看更多&#xff0c;一直按回车。 git reset --hard 目标版本hash值 &#xff0c;注意&#xff1a;这一步操作完后&#xff0c;目标版本之后的代码将全部清掉&am…

python发送各类邮件的主要方法

From: http://www.cnblogs.com/xiaowuyi/archive/2012/03/17/2404015.html python中email模块使得处理邮件变得比较简单&#xff0c;今天着重学习了一下发送邮件的具体做法&#xff0c;这里写写自己的的心得,也请高手给些指点。 一、相关模块介绍 发送邮件主要用到了smtplib和e…

图片保持比例自适应大小;图片保持比例自动充满父元素;图片保持比例充满盒子

需求&#xff1a;经常会有一个div盒子&#xff0c;里面放入一个img图片。需要是图片保持比例&#xff0c;缩放充满div盒子。 思路&#xff1a; 1.必须给父元素设置固定的宽高。2.给img设置 object-fit: scale-down; 属性 object-fit使用学习 先看效果&#xff1a; <!DOCTYP…

Python序列之元组

系列文章目录 Python序列之列表 Python序列之元组 系列文章目录前言一、元组是什么&#xff1f;二、元组操作1.元组的创建&#xff08;1&#xff09;通过()创建。小括号可以省略。&#xff08;2&#xff09;通过tuple()函数创建。&#xff08;3&#xff09;通过生成器推导式创…

Eclipse 下载 开源项目 maven依赖丢失和 Deployment Assembly 丢失

周末下载了最新的jeecg的源码来瞅瞅&#xff0c;但是下载后发现&#xff0c;pom文件中定义的依赖都丢失了。 如下图 上网搜索了一下啊&#xff0c;发现需要先给这个项目这个项目 disable maven nature 然后再添加上再给这个项目添加maven支持。 如下图&#xff1a; 然后再查看项…