当前位置: 首页 > news >正文

c++程序示例:多线程下的实例计数器

我的一个c++项目,有个类,运行过程中会创建多个实例,每个实例独自一个线程。原本程序设置了一个静态、全局的计数器,实际上是所有实例共用,但我希望每个实例管理自己的计数器。最自然的想法,应该设置一个私有变量作为计数器,每个实例操作这个私有变量。但是,程序中有回调函数,不好传递这些个变量。

解决的办法还是设置静态、全局的计数器,但改为集合,每个实例对应一个集合元素。具体代码如下:

定义一个全局的集合,集合元素是一个结构。集合的key,是实例中一个关键的ID值,是唯一的。在实例的构造函数中,即往该集合中添加一个元素。进行计数的目的,不仅仅是为了观察程序运行情况,更重要的是,用于判断是否与设备失联。

一、代码介绍

1、定义

struct ReceiveCount { // 接收计数器long sent_count = 0;//发送计数器long received_count = 0;//接收计数器long last_received_count = 0;//最上一次接收量void reset() {//重置计数器sent_count = 0;received_count = 0;last_received_count = 0;}void send() { sent_count++; }//发送计数void receive() { received_count++; }//接收计数float getLossRate() {//计算接收/发送比率if (sent_count == 0)return 0.0f;return ((float)(sent_count - received_count) / sent_count) * 100.0f;}long getNewReceived() {//最新接收量long new_cnt = received_count - last_received_count;last_received_count = received_count;return new_cnt > 0 ? new_cnt : 0;}long getReceived() { return received_count; }long getSend() { return sent_count; }
};std::map<int, ReceiveCount> g_recivCountMap; // key的类型是int

2、构造

类的构造函数调用此方法

void initRecivCount(int master_id) {reciv_count_mtx.lock();auto it = g_recivCountMap.find(master_id);if (it == g_recivCountMap.end()) {// 不存在:插入新元素(自动调用 ReciveCount 的默认构造)g_recivCountMap.emplace(master_id, ReceiveCount{});} else {// 存在:重置计数器it->second.reset();}reciv_count_mtx.unlock();
}

3、计数

1)向设备发送指令时进行发送计数

  auto it = g_recivCountMap.find(this->master_id);if (it != g_recivCountMap.end()) {it->second.send();}

2)在设备信息回调函数中进行接收计数

    auto it2 = g_recivCountMap.find(master_id);if (it2 != g_recivCountMap.end()) {it2->second.receive();}

4、统计

定期进行统计

auto it = g_recivCountMap.find(this_->master_id);
if (it != g_recivCountMap.end()) {ReceiveCount &counter = it->second;long newRecvNum = counter.getNewReceived();printf("%s : loss data rate: %.2f%%,recive num:(%ld)\n", this_->siimip, counter.getLossRate(), newRecvNum);if (newRecvNum == 0) {no_new++;if (no_new > 10) { // 连续10次没有收到回调消息,即认为已经断连no_new = 0;//此次进行重连。。。}} else {no_new = 0;}
}

二、注意的问题

开发过程中,有一个问题很让我疑惑。就是为什么最新接收量永远等于累计接收量。明明该方法里面有运算

  long getNewReceived() {//最新接收量long new_cnt = received_count - last_received_count;last_received_count = received_count;return new_cnt > 0 ? new_cnt : 0;}

后来AI告诉我,主要是我的调用方法有问题:

ReceiveCount counter = it->second;
long newRecvNum = counter.getNewReceived();

原因是ReceiveCount counter = it->second; 相当于把 g_recivCountMap 中的 ReceiveCount 对象拷贝了一份,在 counter.getNewReceived() 中修改的是 拷贝体的 last_received_count,而 原始 map 中的对象没有更新!(这就跟c#和java的不一样。c#和java的话,couter就应该算是引用类型,修改的都是同一份。)

应该是

ReceiveCount &counter = it->second;
long newRecvNum = counter.getNewReceived();

好奇戒啊。

http://www.dtcms.com/a/333510.html

相关文章:

  • Nginx反向代理与缓存实现
  • 企业级Java项目和大模型结合场景(智能客服系统:电商、金融、政务、企业)
  • 正确维护邵氏硬度计的使用寿命至关重要
  • 【办公类110-01】20250813 园园通新生分班(python+uibot)
  • 量化线性层(42)
  • JavaScript 逻辑运算符与实战案例:从原理到落地
  • JavaScript 中 call、apply 和 bind 方法的区别与使用
  • 技术解读 | 搭建NL2SQL系统需要大模型么?
  • 【Git】Git-fork开发模式
  • 从0开始学习Java+AI知识点总结-15.后端web基础(Maven基础)
  • ARM Cortex-M7 Thread Mode与Handler Mode
  • Android ViewPager2+Fragment viewModelScope问题
  • 在 Vue2 中使用 pdf.js + pdf-lib 实现 PDF 预览、手写签名、文字批注与高保真导出
  • Java零基础笔记18(Java编程核心:Java网络编程—数据通信方案)
  • leetcode 刷题1
  • SysGetVariableString函数
  • 【python实用小脚本-187】Python一键批量改PDF文字:拖进来秒出新文件——再也不用Acrobat来回导
  • 详解 k 近邻(KNN)算法:原理、实践与调优 —— 以鸢尾花分类为例
  • JUC LongAdder并发计数器设计
  • 指针操作:从到*的深度指南
  • JavaWeb开发_Day13
  • Cortex-Debug和openocd之间的关系?如何协同工作?
  • 《人形机器人的觉醒:技术革命与碳基未来》——触觉反馈系统:电子皮肤的概念、种类、原理及在机器中的应用
  • 攻防世界—fakebook(两种方法)
  • docker重启或系统重启后harbor自动启动
  • 深入理解C++正则表达式:从基础到实践
  • ReasonRank:从关键词匹配到逻辑推理,排序准确性大幅超越传统方法
  • Apifox接口测试工具
  • Unity输入系统:旧版Input_System
  • 第四章:大模型(LLM)】06.langchain原理-(3)langchain 数据连接方法