开启辅助访问 切换到宽版

精易论坛

 找回密码
 注册

QQ登录

只需一步,快速开始

用微信号发送消息登录论坛

新人指南 邀请好友注册 - 我关注人的新帖 教你赚取精币 - 每日签到


求职/招聘- 论坛接单- 开发者大厅

论坛版规 总版规 - 建议/投诉 - 应聘版主 - 精华帖总集 积分说明 - 禁言标准 - 有奖举报

查看: 15508|回复: 82
收起左侧

[易源码分享] 易版环形缓冲区 理论上支持多线程

[复制链接]
结帖率:0% (0/2)
发表于 2021-5-15 18:19:07 | 显示全部楼层 |阅读模式   广东省佛山市
分享源码
界面截图: -
是否带模块: -
备注说明: -
本帖最后由 Jy倾恋 于 2021-5-16 14:22 编辑
这个环形缓冲区是基于 http://circularbuffer.codeplex.com/、 http://en.wikipedia.org/wiki/Circular_buffer 修改的
最近自己项目用到的一个缓冲区,理论上支持多线程
在自己的多线程项目测试过,暂时没有问题

下面科普下环形缓冲区在内存里的变化:

环形缓冲区首先从空开始并具有设置的长度;在下图中,是一个7字节的缓冲区:

250px-Circular_buffer_-_empty.svg.png

假设在环形缓冲区的中心写入1(确切的起始位置在环形缓冲区中并不重要):

250px-Circular_buffer_-_XX1XXXX.svg.png

然后,假设将另外两个字节(2&3)添加到环形缓冲区,它们将放在1之后:  

250px-Circular_buffer_-_XX123XX.svg.png

如果删除了两个字节,则环形缓冲区内部的两个最早加入的值将被删除。 环形缓冲区使用FIFO(先进先出)逻辑。 在示例1和2中,第一个进入“环形缓冲区”则第一个被移除,而将3留在缓冲区中。   

Circular_buffer_-_XXXX3XX.svg.png

如果缓冲区有7个字节,则它已经完全占满:

250px-Circular_buffer_-_6789345.svg.png

环形缓冲区的一个特性是,当缓冲区已满并执行后续写入操作时,它将开始覆盖最早的数据。 在当前示例中,添加了两个元素A和B并覆盖了3和4:

250px-Circular_buffer_-_6789AB5.svg.png

最后,如果现在删除了两个字节,则返回的不是3&4而是5&6,因为A&B覆盖了3&4,产生了带有以下内容的缓冲区:  

250px-Circular_buffer_-_X789ABX.svg.png

环形缓冲区使用说明:

环形缓冲区的特性是,在使用环形缓冲区时,不会导致内部数据乱七八糟。 (如果使用了非环形缓冲区,那么在没取一个字节时,就必须对所有字节进行移位。)换句话说,环形缓冲区非常适合作为FIFO(先进先出)缓冲区,而标准缓冲区则适合用作FIFO(先进先出)缓冲区。非环形缓冲区非常适合用作LIFO(后进先出)缓冲区。

对于具有固定最大大小的队列,使用环形缓冲是一种很好的实现策略。如果队列采用最大大小,则环形缓冲区是完全理想的实现;所有队列操作都是固定时间。但是,扩展循环缓冲区需要转移存储器,这是非常耗时和消耗资源的。对于任意扩展的队列,可以首选使用链表方法。

这个是24个字节的环形缓冲区

400px-Circular_Buffer_Animation.gif

当写指针即将到达读指针时(由于微处理器没有响应),缓冲区停止记录击键。 在某些计算机上会发出哔声。

这个排版真累人- .-

QQ截图20210515181027.png


QQ截图20210516140758.png


如果有什么BUG,请自行修复

2021/5/16  源码更新:
[+] 为了大家方便理解新增了Demo


2021/5/15  源码更新:
[!] 修复致命BUG,受影响函数:CircularBufferGetBytes、CircularBufferGetBytesFoIndex、CircularBufferPutToMem、CircularBufferPutBytes
下面为更新后代码:
  
子程序名返回值类型公开备 注
CircularBufferPutToMem整数型 将新的数据复制到环形缓冲区,如果复制的大小超出缓冲区的容量则最早复制的数据将会被覆盖
参数名类 型参考可空数组备 注
lp整数型操作指针
pData整数型
Index整数型pData的偏移值
count整数型需要复制的长度
变量名类 型静态数组备 注
bufferCIRCULARBUFFER_STREAM 
pOld整数型 
srcIndex整数型 
bytesToProcess整数型 
chunk整数型 
如果真 (count ≤ 0)
调试输出 (“计数必须为正。”)
返回 (0)
pOld = _SWAP_CIRCULARBUFFER_STREAM (buffer, lp)
如果真 (buffer.Size + count > buffer.Capacity)
调试输出 (“缓冲区没有足够的容量来放置新项目。”)
_SWAP_CIRCULARBUFFER_STREAM (buffer, pOld)
返回 (0)
InterlockedIncrement (buffer.ref)
srcIndex = Index
bytesToProcess = count
判断循环首 (bytesToProcess > 0)
chunk = 取较小值 (buffer.Capacity - buffer.Tail, bytesToProcess)
RtlMoveMemory (buffer.pBuffer + buffer.Tail, pData + srcIndex, chunk)
判断 (buffer.Tail + chunk = buffer.Capacity)
InterlockedExchange (buffer.Tail, 0)
InterlockedExchangeAdd (buffer.Tail, chunk)
InterlockedExchangeAdd (buffer.Size, chunk)
srcIndex = srcIndex + chunk
bytesToProcess = bytesToProcess - chunk
判断循环尾 ()
CircularBufferFree (lp)
_SWAP_CIRCULARBUFFER_STREAM (buffer, pOld)
返回 (count)
子程序名返回值类型公开备 注
CircularBufferGetBytesFoIndex字节集 从目标缓冲区指定的索引处开始,将指定的字节从对应缓冲区复制取出并删除
参数名类 型参考可空数组备 注
lp整数型操作指针
Index整数型返回的字节集偏移量
count整数型需要复制的长度
变量名类 型静态数组备 注
bufferCIRCULARBUFFER_STREAM 
pOld整数型 
bytesCopied整数型 
dstIndex整数型 
chunk整数型 
arybyte字节集 
pData整数型 
如果真 (Index < 0)
调试输出 (“指定偏移量必须为正 ”)
返回 ({ })
如果真 (count < 0)
调试输出 (“指定的长度必须为正。”)
返回 ({ })
pOld = _SWAP_CIRCULARBUFFER_STREAM (buffer, lp)
如果真 (count > buffer.Size)
调试输出 (“环形缓冲区的内容不足以进行读取/读取操作。”)
_SWAP_CIRCULARBUFFER_STREAM (buffer, pOld)
返回 ({ })
InterlockedIncrement (buffer.ref)
bytesCopied = 0
dstIndex = Index
arybyte = 取空白字节集 (count)
pData = 取指针_字节集型 (arybyte)
判断循环首 (count > 0)
chunk = 取较小值 (buffer.Capacity - buffer.Head, count)
RtlMoveMemory (pData + dstIndex, buffer.pBuffer + buffer.Head, chunk)
判断 (buffer.Head + chunk = buffer.Capacity)
InterlockedExchange (buffer.Head, 0)
InterlockedExchangeAdd (buffer.Head, chunk)
InterlockedExchangeAdd (buffer.Size, -chunk)
dstIndex = dstIndex + chunk
bytesCopied = bytesCopied + chunk
count = count - chunk
判断循环尾 ()
CircularBufferFree (lp)
_SWAP_CIRCULARBUFFER_STREAM (buffer, pOld)
返回 (arybyte)


i支持库列表   支持库注释   
spec特殊功能支持库


源码回复可见,不喜勿喷
CircularBuffer.rar (340.88 KB, 下载次数: 111)

点评

使用自定义数据类型会让你的缓冲区快不起来   广东省深圳市  发表于 2022-11-30 15:02
看不懂的不会自学英语或者自己百d翻译吗... 饭都放你面前了还得别人喂吗   广东省潮州市  发表于 2022-3-31 18:13
英文不好,看不懂   广东省潮州市  发表于 2021-8-24 03:04
全是英文,这谁看得懂   湖南省岳阳市  发表于 2021-6-12 17:18
有人可能知道怎么用这个缓冲区,在通信程序中,经常使用环形缓冲器作为***结构来存放通信中发送和接收的***,主要作用是用来分包和组包的   广东省佛山市  发表于 2021-5-16 01:31

评分

参与人数 10好评 +7 精币 +18 收起 理由
fjgh + 1 + 2 支持开源~!感谢分享
木小果 + 3 支持开源~!感谢分享
BUG508 + 1 + 2 感谢大佬开源,这个环形缓冲区是真的好用,新技能已get√
AI优 + 1 支持开源~!感谢分享
小言T默默 + 2 支持开源~!感谢分享
哆啦恶梦 + 1 + 2 感谢分享,很给力!~
司徒西 + 1 + 2 支持开源~!感谢分享
广二爷xxoo + 1 此处应该有鼓励~
吃泡面加不起蛋 + 1 + 1 显示日志也用到这个,一共500个日志,循环更新
易语言资源网 + 1 + 3 支持开源~!感谢分享

查看全部评分


本帖被以下淘专辑推荐:

结帖率:100% (7/7)

签到天数: 9 天

发表于 2022-12-1 15:45:54 | 显示全部楼层   广东省佛山市
支持开源~!感谢分享
回复 支持 反对

使用道具 举报

结帖率:98% (41/42)

签到天数: 8 天

发表于 2022-5-1 14:59:40 | 显示全部楼层   河北省邢台市
谢谢开源 感谢你
回复 支持 反对

使用道具 举报

签到天数: 3 天

发表于 2022-4-2 20:07:35 | 显示全部楼层   广东省汕尾市
66666666666666666666666666666666666
回复 支持 反对

使用道具 举报

结帖率:100% (1/1)

签到天数: 9 天

发表于 2022-3-22 14:10:02 | 显示全部楼层   陕西省宝鸡市
支持开源~!感谢分享
回复 支持 反对

使用道具 举报

发表于 2022-3-19 17:39:42 | 显示全部楼层   陕西省西安市
666666666666666666666666666666666666666666666666
回复 支持 反对

使用道具 举报

发表于 2022-3-14 23:50:39 | 显示全部楼层   江西省赣州市
支持开源~!感谢分享
回复 支持 反对

使用道具 举报

结帖率:100% (7/7)

签到天数: 2 天

发表于 2022-3-6 02:33:27 | 显示全部楼层   广东省潮州市
很厉害,收下研究了
回复 支持 反对

使用道具 举报

结帖率:67% (2/3)
发表于 2022-3-3 09:58:40 | 显示全部楼层   四川省成都市
支持开源~!感谢分享
回复 支持 反对

使用道具 举报

结帖率:40% (2/5)
发表于 2022-2-18 11:50:08 | 显示全部楼层   湖南省长沙市
多多回复,多多赞币
回复 支持 反对

使用道具 举报

结帖率:82% (27/33)
发表于 2022-2-12 04:00:54 | 显示全部楼层   上海市上海市
共同努力,共同进步
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 注册

本版积分规则 致发广告者

发布主题 收藏帖子 返回列表

sitemap| 易语言源码| 易语言教程| 易语言论坛| 易语言模块| 手机版| 广告投放| 精易论坛
拒绝任何人以任何形式在本论坛发表与中华人民共和国法律相抵触的言论,本站内容均为会员发表,并不代表精易立场!
论坛帖子内容仅用于技术交流学习和研究的目的,严禁用于非法目的,否则造成一切后果自负!如帖子内容侵害到你的权益,请联系我们!
防范网络诈骗,远离网络犯罪 违法和不良信息举报电话0663-3422125,QQ: 793400750,邮箱:wp@125.la
Powered by Discuz! X3.4 揭阳市揭东区精易科技有限公司 ( 粤ICP备12094385号-1) 粤公网安备 44522102000125 增值电信业务经营许可证 粤B2-20192173

快速回复 返回顶部 返回列表