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

2 篇文章 0 订阅
1 篇文章 0 订阅

前言

针对传统超市购物车结账排队时间长、付款效率低的问题,提出了一种更符合现代社会人们购物方式-基于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)
)

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

 

 

 

  • 107
    点赞
  • 125
    收藏
    觉得还不错? 一键收藏
  • 107
    评论
【资源说明】 1.项目代码均经过功能验证ok,确保稳定可靠运行。欢迎下载食用体验! 2.主要针对各个计算机相关专业,包括计算机科学、信息安全、数据科学与大数据技术、人工智能、通信、物联网等领域的在校学生、专业教师、企业员工。 3.项目具有丰富的拓展空间,不仅可作为入门进阶,也可直接作为毕设、课程设计、大作业、初期项目立项演示等用途。 4.当然也鼓励大家基于此进行二次开发。在使用过程中,如有问题或建议,请及时沟通。 5.期待你能在项目中找到乐趣和灵感,也欢迎你的分享和反馈! 【项目介绍】 C++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zipC++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zipC++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zipC++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zipC++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zipC++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zipC++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zipC++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zip C++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zip C++开发基于linux多线程个人图片浏览HTTP服务器源码+超详细注释.zip
### 回答1: Linux下的多线程编程可以使用C/C++语言实现。C/C++语言提供了一些多线程编程的库,如pthread库、OpenMP库、Boost库等。其中,pthread库是Linux下最常用的多线程编程库,它提供了一系列的API函数,可以用来创建、管理和同步线程。在C/C++语言中,可以使用pthread_create()函数创建线程,使用pthread_join()函数等待线程结束,使用pthread_mutex_lock()和pthread_mutex_unlock()函数实现线程间的互斥访问等。同时,C++11标准也提供了一些多线程编程的支持,如std::thread类、std::mutex类等,可以方便地实现多线程编程。 ### 回答2: Linux下的多线程编程是指在Linux系统下使用多个线程来执行不同的任务,从而提高程序的运行效率和响应速度。 C/C++Linux下最常用的编程语言之一,也是多线程编程的主要语言。实现多线程编程可以使用线程库,其中最常用的是pthread库。 Pthread库是Linux下的开放式多线程库,它允许程序员使用标准的POSIX线程接口来创建、终止、同步和管理线程。使用Pthread库可以很方便地进行多线程编程,其中主要包括以下几个方面。 1. 创建和启动线程:使用pthread_create函数来创建和启动线程,该函数需要传递线程ID、线程属性和线程函数等参数。 2. 同步线程:使用pthread_join函数来等待一个线程结束,以便获取线程的返回值。使用pthread_mutex和pthread_cond等函数来进行线程同步。 3. 线程控制使用pthread_cancel函数来取消线程,使用pthread_exit函数来终止线程。 4. 共享变量:在多个线程之间共享变量时,需要使用pthread_mutex和pthread_cond等函数来控制并发访问。 在进行多线程编程时,需要注意一些问题,如线程安全、死锁等问题。不同的线程对共享资源的读写需要使用同步机制,避免竞争和冲突。此外,要注意避免死锁,即多个线程互相等待对方释放资源,造成程序无法正常运行。 总之,Linux下的多线程编程是一项非常重要的技术,在实际开发中应用广泛。使用C/C++编写多线程程序,需要熟悉线程库的使用方法,掌握线程的创建、同步、控制和共享等技术,以保证程序的稳定性和运行效率。 ### 回答3: Linux是一种开源的操作系统,其多线程编程能力是其强大之处之一。当我们需要编写一个高性能、高并发的程序时,多线程编程无疑会是一个很好的选择。 在Linux下,C/C++是最常用的编程语言之一,也是多线程编程的重要语言之一。在C/C++中编写多线程程序主要依赖于pthread库。pthread库提供了一套多线程API,可以很方便的创建和管理线程。 使用pthread库创建线程需要以下步骤: 1. 包含pthread库头文件: #include <pthread.h> 2. 定义线程函数: void *thread_func(void *arg){ //线程执行的代码 } 3. 创建线程: pthread_t tid; pthread_create(&tid, NULL, thread_func, NULL); 4. 等待线程结束: pthread_join(tid, NULL); 以上代码片段就创建了一个新线程,并在新线程中执行了thread_func函数。pthread_create函数的第一个参数为线程ID,第二个参数为线程属性,一般使用NULL,第三个参数为线程函数,第四个参数为线程函数的参数。 多线程编程需要注意以下几点: 1. 线程安全:多个线程同时操作同一个共享资源,需要确保操作的正确性和安全性。 2. 线程同步:使用锁、互斥量等机制保证线程之间的同步。 3. 线程调度:多个线程之间需要进行调度,需要注意线程优先级的设置。 总之,在Linux使用C/C++进行多线程编程是一项非常有用的技能。在实际开发中,需要结合具体的场景和需求,通过选择合适的多线程编程模型、算法和数据结构来实现高效、高性能的程序。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 107
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值