基于Linux C++多线程服务器 + Qt上位机开发 + STM32 + 8266WIFI的智慧无人超市

news2025/1/12 6:09:00

前言

针对传统超市购物车结账排队时间长、付款效率低的问题,提出了一种更符合现代社会人们购物方式-基于RFID的自助收银系统。习惯了快节奏生活的人们都会选择自助收银机结账,理由显而易见:自助收银机结账很方便,几乎不用排队,也不用近距离和收银员接触,在防疫时期特别感觉安心。而且自助结账对每件物品的售价更是一次清晰地核对,最终需支付合计购物支出自己也更加清晰明了;这两年来,越来越多的智能设备应用在我们的生活领域里,为我们的生活提供了很多智能和便利。自助收银机从几年前就陆续涌入到各地商场、超市、便利店,自去年疫情发生后自助收银的需求比例更是呈直线上升趋势。自助收银机的启用,不仅节约了超市的人力开支成本,也从根本上提升了超市的购物支付效率,在这个快节奏的社会里,智能自助收银机也从根本上提升了超市等

 

基于Linux C++多线程服务器 + Qt上位机开发 + STM32 + 8266WIFI的智慧无人超市项目

 

 

技术栈+硬件选型

Linux c++应用编程;

Linux socket编程,多线程编程,实时信号(线程通信)

Qt/C++ 客户端开发;

qml(QtLocation)与c++交互 安卓开发;

Mysql 数据存储;

C语言下位机开发;

stm32c8t6 下位机(便宜,够用,市面价格10r);

RC522(RFID模块 SPI协议) 与白卡通信 获取卡号;

DTH11 温湿度采集模块(单总线协议,市面价格 4r);

sg90 舵机模块(PWM协议 市面价9r );

8226 01-s WIFI模块(uart协议 市面价格5r) 连接 C++ 服务器;

蜂鸣器  是为了有白卡与RFID交互声音

GY-NEO6MV2 GPS模块(uart协议 市面价18)

 

 

总设计流程

本次设计的->基于RFID的自助收银系统->设计主要支持的功能和程序如下:
本项目一共有五个程序,Linux C++服务器,Qt管理员端,Qt客户结算端,qml安卓端,单片机下位机端,客户端程序都有服务器断开自动重连(单片机没有,有佬会的话,我想请教一下)

Linux C++服务器:

       基于 socket 接口编写server服务端程序,监听8888端口,然后创建Mysql数据库连接,开始监听。

     面向对象程序设计中最重要的一个概念是继承,所以我编写了一个基类mythread,他有一个纯虚函数,参数为一个定义的一个参数类,包含了数据库封装好的对象,需要服务的客户端套接字,还有连接的客户端网络的信息结构体,当socket客户端成功接入时,取出该客户端套字和网络信息结构体和主函数的数据库对象来填充参数类,然后服务器端根据客户端发过来的第一个消息判断该客户端是上述哪个客户端,然后创建相应的派生类,填入该参数类,然后让mythread指针去指向这个子类对象,发生动多态,此时运行阶段时才确定函数的入口地址,执行派生类中的继承父类的已经实现的纯虚函数,此时派生类创建一个线程绑定线程处理函数去服务处理该socket套接字传过来的消息。绑定不同的线程处理函数服务不同的客户端程序。一共有四种线程处理函数服务上述四种客户端程序,由四个不同的基类去绑定。

    本项目的一个最大的特点,难点,单片机只负责发送卡号给服务器,本项目单片机传过来的卡号有以下走向,注册商品,注册会员,结账,商品入购物车,那么我该怎么知道该卡号是用来做什么呢,当时困扰了几天,然后想到Qt的信号与槽机制,联想到Linux 也有实时信号,还可携带参数,该信号是事件发生时对进程的通知机制,也可以把它称为软件中断,Linux 内核定义了 31 个不同的实时信号,信号编号范围为 34~64,我直接拿来做线程通信,所以现在怎样把他们联系起来呢,当时我想到互斥啥的想法感觉不好实现,然后我就想到了一个方法,就是设置标志位,我设立了一个全局变量 int RES=34,让RES的默认值为34,34很熟悉吧,信号编号范围为 34~64,当单片机收到卡号发给服务器时,服务器直接调用信号发送函数,此时全局变量标志位为34,所以直接发送34这个实时信号,然后触发中断执行这个信号绑定处理函数。我们一共有四个地方需要用到卡号,资源只有一个,所以当需要执行某个用到卡号的操作时,我先判断该RES的值,如果该值等于34,代表该读卡器为空闲,我就更改RES为 35 ,然后下次单片机发过来卡号,我还是直接调用发送信号,此时标志位为35 所以此时执行 35 的信号处理函数,执行完函数,需要将RES 置为默认值 34,释放资源,如果当请求资源时RES不等于 34 代表 读卡器正在被占用,此时回复客户端一个读卡器正忙,以此类推,绑定四个信号处理函数,处理这四个操作请求,一定要释放改为34。我们添加商品到购物车,如果一直点击按钮获取来获取资源就会显得很笨拙,所以默认的 34 信号处理函数为添加商品入购物车,在没有人改变标志位的情况下,就是执行商品入购物车。

   本项目,因为多个线程对数据进行增删改查,存在竞争冒险,所以在执行数据库增删改操作时,我加入条 Mysql 事务操作语句,事务是一个原子操作,执行增删改操作前 开始事务,执行结束,提交事务。

   总结 : 单片机的所有数据全部转发给服务器,服务器跟据消息的种类,标志位,进行处理后,分发给指定客户端,完成一系列操作。

主循环如下:

 while (1)
   {
      myret=server.client_socket();
      myret.my_sql=sql_typ;

      std::cout<<"new connect !!"<<endl;
      std::string str=server.readbuf();

      int num; 
      istringstream a(str);
      a >> num;

    
     switch (num)
  {
     {
     case 100001:   
    
         Mythread *android_thread=new androidthread; 
         android_thread->thread_start(myret);
         std::cout<<"安卓客户端连接成功"<<std::endl; 
     }
         break;
     case 100101:
     {
          Mythread  *admin_thread=new adminthread;   
          admin_thread->thread_start(myret);
          std::cout<<"PC客户端连接成功"<<std::endl;
     }
         break;
     case 100111:
     {
          Mythread  *cust_thread=new custthread;    
          cust_thread->thread_start(myret);
          std::cout<<"ARM客户端连接成功"<<std::endl;
     } 
         break;
     case 101001:
     {

          Mythread   *mcu_thread=new mcuthread;    
          mcu_thread->thread_start(myret);
          std::cout<<"STM32客户端连接成功"<<std::endl;
     }  
         break;
     default:
         std::cout<<"未知的错误"<<std::endl;   
         break;
     }

        
   }
   


 

Qt管理员端:

Qt管理员端具有的功能,注册商品,注册会员,充值,查看销售记录,日志。

首先连接服务器成功,自动发指令给服务器,服务器从数据库取出数据发给客户端,初始化,商品,会员,服务器上有一个文本文件记录销售记录,我给服务器文本大小做了一个限制,如果大小大于1M清空文件,清除销售记录,客户端可以通过点击按钮发送一个指令,获取文本内容显示销售记录,日志的话就是从服务器运行阶段开始,对会员充值,会员销毁,商品添加,商品删除,都会直接记录,服务器退出自动销毁。

Qt客户结算端:

Qt客户结算端具有的功能:添加商品入购物车,结算,显示从服务器端获取的温湿度数据。

商品入库取出价格,然后相加,点击结算按钮将总价格发给服务器,然后服务器判断标志位,如果读卡器被占用则取消结账,反之。此时标志位改变,下次刷的卡将充当会员卡进行数据比对,余额不足则返回数据给客户端,反之。执行完毕释放资源置为34。

qml安卓端:

qml安卓端具有的功能:地图显示当前手机与MCU的位置和距离,在售商品查询,购买记录查询,姓名号码登录。

这个安卓端其实有点画蛇添足的意思,我就是想炫耀一下我的GPS模块,然后地图显示当前手机GPS数据与MCU的GPS模块的距离和位置,功能太单调了,我就加了一个在售商品查看功能,然后给Mysql添加了1000大小的varcahr字段,存储当前用户购买记录,加了一个登录界面。

   qml的教程挺少,之前学过一遍,没有及时巩固,当时写这个qml真的炸裂,很多坑。想入门qml也简单,学一下qml与C++交互,信号与槽,函数互调。qml界面的话让gpt去写,百度CV。

STM32单片机开发:

stm32具体的功能:stm32c8t6主控芯片,DTH11温湿度采集发送给服务器客户端显示,sg90舵机模拟开门,GY-NEO6MV2 GPS 获取GPS,8226 01-s 与tcp 服务器数据交互,RC522与白卡交互,蜂鸣器提示刷卡成功。

32这调试是最恶心的,一般调试是直接通过串口打印到电脑,但是串口用来初始化8266了,问题就是这个8266,当时连接服务器一直连接不上,我就去找原因,有供电原因,还有at指令的原因,供电最好直接从32vcc上引出来,因为我是根据客户端程序连接成功后根据发过来的第一条数据来创建对应线程服务,不知道为啥这个32程序按复位键的话8266没有从第一条指令开始运行,然后就创建不了对应线程服务,只能断电,然后在重新烧录一次,调试巨麻烦。

GPS的话也是串口通信,重新初始化一个串口资源就好了,这个信号很差必须在阳台上,有条件的还是买好一点的吧我采集数据还有抱着一大堆线接个充电宝在阳台调试。gps数据有个NMEA协议,需要对数据解析出经纬度,有很多类型数据,最重要的一条就是包含经纬度的 

 

"$GPGLL,2547.35222,N,11306.12283,E,111129.00,A,A*6D";

这是当时当时在阳台调试出来的,这里我偷了个小懒,因为c语言字符串处理很鸡肋,所以我直接在32这里过滤出这条消息,一整条发给服务器,然后服务器发给Qt客户端,让Qt的QString去解分割分析处理,分分钟的事情,封装好的库就是简单

 

f4eff0cf6b7a4b8fa8a8ca56f53c46cc.png

很简单吧!

8266的话我初始化很随意,快准狠,直接配置tcp,连接WiFi然连接服务器,哈哈。

e5ae932d216c4862aed5ce3270a5ab51.png

温湿度舵机什么的没什么好讲,一个单总线写按时序拉低拉高电平就行了,一个设置指定占空比。RC522的话,驱动很复杂,我水平不太行,写不出来,直接调厂家的库了。

主循环就这样了

3af14c79a4c84857a7c467f3d64038cb.png

 

 

最后留言:

本项目一共写了半个月,遇到很多坑,当时地图传的坐标经纬度传犯了,调试了一下午才发现了这个问题,还有Linux 实时信号 是一个软件中断嘛,然后当时server的while循环的accept直接导通解除阻塞了,程序直接崩溃。查了很多资料才知道了,最后加一个goto语句哈哈就解决了。

ret_client  Myserver::client_socket()
{
reboot:
     ret_client ret;
  
     m_client_socket= accept(m_socket,(struct sockaddr *)&ret.client_struct,&len);

     if(m_client_socket==-1)
     {
       goto reboot;
     }

    ret.client_socket=m_client_socket;
    return ret;

}

这是sql语句啊,本人挺懒的,全用varchar了。

CREATE DATABASE shopping;

CREATE TABLE users(
id VARCHAR(20) unique key,
name VARCHAR(50),
phone VARCHAR(15) unique key,
balance VARCHAR(25),
text VARCHAR(1000)
)

CREATE TABLE me(
pid VARCHAR(20) unique key,
pname VARCHAR(50),
price VARCHAR(15),
brand VARCHAR(25)
)

需要源码的哥们三连加评论邮箱,直接发邮箱

 

 

 

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

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

相关文章

【java9】java9新特性之改进进程管理API

Java9在改进进程管理API方面&#xff0c;主要的更新在于引入了ProcessHandle接口&#xff0c;该接口代表了一个本地进程&#xff0c;并提供了许多与进程相关的信息和操作。 通过ProcessHandle接口&#xff0c;开发者可以获取进程的ID、父进程、子进程、进程命令行参数、进程状…

QT——简易计算机(从0开始)

目录 一、题目描述&#xff1a; 二、创建工程&#xff1a; 三、UI界面设计&#xff1a; 四、程序编写&#xff1a; 五、总程序&#xff1a; 六、windows可执行文件 七、实现效果 一、题目描述&#xff1a; 创建一个简单的图形用户界面(GUI),包括一个文本框用于显示计算结…

ctfshow-web入门-102

这个题我想记录一下&#xff0c;主要是这个方法属实是有点惊艳到我了。故而进行记录&#xff0c;也为了方便大家阅读理解。 看题目&#xff0c;根据题目我写一下我的分析&#xff1a; $_POST传入一个v1&#xff0c;$_GET传入一个v2&#xff0c;一个v3。 赋值符号 优先级高于…

谷粒商城实战(021 业务-订单模块-页面设计)

Java项目《谷粒商城》架构师级Java项目实战&#xff0c;对标阿里P6-P7&#xff0c;全网最强 总时长 104:45:00 共408P 此文章包含第262p-第p266的内容 介绍 所需的页面 设计页面 新增域名 路径带/static的都到/usr/share/nginx/html文件夹下去找 其他动态请求的都负载…

2024年第二十一届 五一杯 (C题)大学生数学建模挑战赛 | 多目标优化问题,深度学习分析 | 数学建模完整代码解析

DeepVisionary 每日深度学习前沿科技推送&顶会论文&数学建模与科技信息前沿资讯分享&#xff0c;与你一起了解前沿科技知识&#xff01; 本次DeepVisionary带来的是五一杯的详细解读&#xff1a; 完整内容可以在文章末尾全文免费领取&阅读&#xff01; 首先&…

Mac M2 本地下载 Xinference

想要在Mac M2 上部署一个本地的模型。看到了Xinference 这个工具 一、Xorbits Inference 是什么 Xorbits Inference&#xff08;Xinference&#xff09;是一个性能强大且功能全面的分布式推理框架。可用于大语言模型&#xff08;LLM&#xff09;&#xff0c;语音识别模型&…

【已解决】pandas读excel中长数字变成科学计数法的问题

pandas 读excel中的长数字时&#xff0c;即使excel中已经设置为文本&#xff0c;读进df后也会自动变成科学计数法。 在日常的数据分析和处理工作中&#xff0c;Excel和pandas是数据分析师们不可或缺的得力助手。然而&#xff0c;在使用pandas读取Excel文件时&#xff0c;我们有…

基于残差神经网络的汉字识别系统+pyqt前段界面设计

研究内容: 中文汉字识别是一项具有挑战性的任务&#xff0c;涉及到对中文字符的准确分类。在这个项目中&#xff0c;目标是构建一个能够准确识别中文汉字的系统。这个任务涉及到数据集的收集、预处理、模型训练和评估等步骤。尝试了使用残差神经网络&#xff08;ResNet&#x…

Rust Rocket创建第一个hello world的Web程序 Rust Rocket开发常用网址和Rust常用命令

一、Rust Rocket简介 Rust Rocket 是一个用 Rust 语言编写的 Web 应用框架&#xff0c;它结合了 Rust 的安全性和性能优势&#xff0c;以及 Web 开发的便利性。以下是 Rust Rocket 框架的一些优点&#xff1a; 安全性&#xff1a;Rust 是一种注重安全性的编程语言&#xff0c;…

【前端探索者:从零到精通的Web前端实战专栏】

🚀 在这个代码编织梦想的时代,Web前端作为互联网的颜值担当,正以日新月异的速度重塑数字世界。想要在前端江湖里游刃有余,你需要的不仅仅是一把锋利的剑,更是一套完整的武功秘籍!今天,我们就为你揭开【Web前端】专栏的神秘面纱,带你从菜鸟到大神,一飞冲天! 📚 专栏…

Text-to-SQL小白入门(12)Awesome-Text2SQL开源项目star破1000

项目介绍 项目地址 23年9月份刚开源这个项目&#xff0c;大半年过去了&#xff0c;star数终于破1000啦&#xff0c;决定在知乎更新一下内容&#xff0c;看看内容变化&#xff0c;知乎有上当时项目介绍的链接&#xff1a;追光者&#xff1a;Text-to-SQL小白入门&#xff08;六&…

【Qt QML】Frame组件

Frame&#xff08;框架&#xff09;包含在&#xff1a; import QtQuick.Controls继承自Pane控件。用于在可视框架内布局一组逻辑控件。简单来说就是用来包裹和突出显示其他可视元素。Frame不提供自己的布局&#xff0c;但需要自己对元素位置进行设置和定位&#xff0c;例如通过…

Spring Boot Admin

概述 Spirng Boot Admin 登录页面 Spring Boot Admin是一个用于管理Spring Boot应用的监控工具,它允许你查看和管理多个Spring Boot应用实例。用于应用信息进行界面化的展示&#xff0c;常常辅助我们开发人员快速查看服务运行状态在微服务架构中&#xff0c;Spring Boot Admin通…

【linux】进程(深入理解linux进程状态)

开始之前先说一个与本文无关的小知识&#xff0c;chdir命令可以更改当前进程的工作目录哦。 目录 linux具体进程状态&#xff1a;R && S&#xff1a;T && t&#xff1a;D&#xff1a;僵尸进程 && 孤儿进程&#xff1a; OS的理论线&#xff1a;运行&…

JMeter性能压测脚本录制

第一步&#xff1a;电脑打开控制面板设置代理服务器 第二步&#xff1a;jmeter的测试计划添加一个HTTP&#xff08;S&#xff09;脚本记录器 在脚本记录器里配置好信息&#xff0c;然后保存为脚本文件&#xff08;.*表示限定&#xff09; 此方框内容为项目地址&#xff08;可改…

如何安装cuda版本的torch-sparse和torch-scatter

安装对应cuda版本的torch&#xff0c;确保cuda可用 使用nvidia-smi查看cuda版本&#xff0c;我的是11.4&#xff0c;然后就找到pytorch历史版本&#xff0c;页面搜索cuda 11.4&#xff0c;没搜到&#xff0c;继续往小版本搜&#xff0c;搜到cuda 11.3&#xff0c;果断安装&…

日拱一卒,月进一步(10)

303. 区域和检索 - 数组不可变 - 力扣&#xff08;LeetCode&#xff09; 动态规划~ 前缀和 最朴素的思想是存储数组nums的值&#xff0c;每次调用sumRange时&#xff0c;通过循环的方法计算数组nums从下标i到下标j的元素和&#xff0c;需要计算j-i1个元素的和。由于每次检索…

Sobel算法:边缘提取的原理与实践【基于python、C++基于opencv的代码实现!!】

Sobel算法&#xff1a;深入解析边缘检测的原理与实现 在图像处理领域&#xff0c;边缘检测是一项至关重要的任务。其中&#xff0c;Sobel算法以其高效和稳定的性能&#xff0c;成为边缘检测中的常用方法之一。本文将深入解析Sobel算法的原理与实现&#xff0c;带您了解如何通过…

02_机器学习算法_基于XGBoost的分类预测

1. XGBoost 算法 1.1 XGBoost 的介绍 XGBoost是2016年由华盛顿大学陈天奇老师带领开发的一个可扩展机器学习系统。严格意义上讲XGBoost并不是一种模型,而是一个可供用户轻松解决分类、回归或排序问题的软件包。它内部实现了梯度提升树(GBDT)模型,并对模型中的算法进行了诸多…

通过先序和中序求后序(通过后序和中序求先序)

一、通过先序和中序求后序 二、通过后序和中序求后序 1、利用后序和中序的特点来求出二叉树 2、后序输出二叉树 如图&#xff0c;先给了一个后序和中序&#xff0c;后序的最后一个为根节点 然后在中序中查找相同的元素&#xff0c;在这个元素左边的为左子树&#xff0c;右…