基于物联网设计的智能储物柜(4G+华为云IOT+微信小程序)

一、项目介绍

在游乐场、商场、景区等人流量较大的地方,往往存在用户需要临时存放物品的情况,例如行李箱、外套、购物袋等。为了满足用户的储物需求,并提供更加便捷的服务体验,当前设计了一款物联网智能储物柜。

该智能储物柜通过与华为云物联网服务器的连接,实现了数据的传输和管理,让用户可以通过微信小程序轻松查看储物柜的可用状态和选择合适的储物柜进行解锁。通过使用4G联网模块E29 Cat-1,储物柜实现了稳定的网络连接,确保了用户可以随时随地查看和使用储物柜。

主控芯片采用STC12C5A60S2,具有低功耗、高性能和丰富的外设接口等特点,能够满足储物柜的控制需求。而储物柜的锁则采用了电磁锁,能够快速、安全地实现储物柜的解锁和上锁操作。

用户在解锁储物柜后,会扣除相应的押金(预付金),然后根据使用时间进行计费。用户可以随时存取自己的物品,并且在使用完成后可以通过微信小程序完成订单结算,使整个使用过程更加便捷和透明。

这款4G智能储物柜为用户提供了一种灵活、安全、便捷的储物解决方案,能够有效满足人们日常生活中的储物需求,并提供良好的用户体验。

智能储物柜支持的主要功能:

(1)远程查看储物柜状态:用户可以通过微信小程序实时查看储物柜的可用状态,包括储物柜数量、剩余可用储物柜数量等信息。这使得用户在选择储物柜时可以提前了解储物柜的可用性,避免不必要的等待。

(2)指定储物柜解锁:用户可以在微信小程序上选择需要使用的储物柜,并进行解锁操作。当用户选择指定储物柜后,储物柜会自动解锁,方便用户存放物品。

(3)押金扣除和计费:在解锁储物柜时,系统会自动扣除用户的押金(预付金)。智能储物柜根据使用时间进行计费,用户只需根据实际使用时间支付相应费用。

(4)存取物品:一旦用户成功解锁储物柜,他们就可以将物品存放或取出。用户可以随时存取自己的物品,并享受到安全可靠的储物体验。

(5)临时解锁功能:如果用户需要频繁存取物品,储物柜提供了临时解锁功能。用户可以选择设置一段时间内的临时解锁,无需重复进行解锁操作,方便快捷。

(6)订单结算:用户在使用完成后,可以通过微信小程序完成订单结算。系统会根据实际使用时间和费率进行计算,并生成相应的费用清单供用户支付。

整体系统流程框架图:

image-20230925153026023

做好的柜子效果如下: (柜子不带屏幕的,全部操作在微信小程序上进行)

image-20230925151722754

二、软硬件选型

2.1 硬件选型

智能储物柜所需的硬件主要包括下面几个部分:

(1)主控芯片:采用STC12C5A60S2主控芯片,具有低功耗、高性能和丰富的外设接口,能够满足储物柜的控制需求。

(2)4G联网模块:采用华为云IOT提供的4G联网模块E29 Cat-1,确保储物柜具备稳定的网络连接功能,实现远程数据传输和管理。

(3)电磁锁(每个小柜子一个电磁锁):储物柜采用电磁锁作为解锁和上锁的装置,能够快速、安全地实现储物柜的开关操作。电磁锁具有较高的稳定性和安全性。

(4)电源系统:储物柜需要一个可靠的电源系统来供电,以确保储物柜的正常运行。

(5)外壳和结构组件:储物柜外壳。

采用主控板:

image-20230925150325217

采用的电磁锁:

image-20230925150848200

采用的柜子:

image-20230925151803556

2.2 软件选型

软件包含了4个部分。

(1)设备端的程序开发,智能储物柜设备端的程序。

(2)华为云IOT物联网服务器的配置。设备注册、消息转发。

(3)智能锁的后台云服务器搭建。

(4)小程序、公众号程序开发。

在这里插入图片描述

三、华为云物联网服务器部署

3.2 开通物联网服务

地址: https://www.huaweicloud.com/product/iothub.html

image-20230801151136153

点击总览,查看接入信息。 我们当前设备准备采用MQTT协议接入华为云平台,这里可以看到MQTT协议的地址和端口号等信息。

image-20230925155200396

image-20230925155233810

总结:

端口号:   MQTT (1883)  
接入地址: e244e6efb9.st1.iotda-device.cn-north-4.myhuaweicloud.com

根据域名地址得到IP地址信息:

Microsoft Windows [版本 10.0.19045.3448]
(c) Microsoft Corporation。保留所有权利。

C:\Users\11266>ping e244e6efb9.st1.iotda-device.cn-north-4.myhuaweicloud.com

正在 Ping e244e6efb9.st1.iotda-device.cn-north-4.myhuaweicloud.com [117.78.5.125] 具有 32 字节的数据:
来自 117.78.5.125 的回复: 字节=32 时间=41ms TTL=94
来自 117.78.5.125 的回复: 字节=32 时间=44ms TTL=94
来自 117.78.5.125 的回复: 字节=32 时间=43ms TTL=94
来自 117.78.5.125 的回复: 字节=32 时间=42ms TTL=94

117.78.5.125 的 Ping 统计信息:
    数据包: 已发送 = 4,已接收 = 4,丢失 = 0 (0% 丢失),
往返行程的估计时间(以毫秒为单位):
    最短 = 41ms,最长 = 44ms,平均 = 42ms

C:\Users\11266>

image-20230925155314730

MQTT协议接入端口号有两个,1883是非加密端口,8883是证书加密端口,单片机无法加载证书,所以使用1883端口比较合适。 接下来的ESP8266就采用1883端口连接华为云物联网平台。

3.3 创建产品

(1)创建产品

点击产品页,再点击创建产品。

image-20230925155450476

(2)填写产品信息

根据自己产品名字填写。

image-20230925155543528

(3)产品创建成功

image-20230925155557999

(4)添加自定义模型

产品创建完成之后,点击进入产品详情页面,翻到最下面可以看到模型定义。

image-20230925155634106

这个模型就是定义自己设备接下来需要向服务器上传那些数据类型。根据自己的数据类型进行编写。

先点击自定义模型。

image-20230925155710677

再创建一个服务ID。

image-20230925155738113

接着点击新增属性。

image-20230925155908114

设置属性。

image-20230925155933814

属性添加成功。

image-20230925155956728

3.4 添加设备

产品是属于上层的抽象模型,接下来在产品模型下添加实际的设备。添加的设备最终需要与真实的设备关联在一起,完成数据交互。

(1)注册设备

image-20230925160050686

(2)根据自己的设备填写

设备标识码、密码这些根据自己情况认真填写。

image-20230925160152785

(3)保存设备信息

创建完毕之后,点击保存并关闭,得到创建的设备密匙信息。该信息在后续生成MQTT三元组的时候需要使用。

image-20230925160232066

创建之后,得到的设备信息如下:

{
    "device_id": "65113d05a559fd7cd41435f8_lock1",
    "secret": "12345678"
}

(4)设备创建完成

可以点击设备进入到设备详情页面。

image-20230925160613806

3.5 MQTT协议主题订阅与发布

(1)主题订阅格式

帮助文档地址:https://support.huaweicloud.com/devg-iothub/iot_02_2200.html

image-20230801151350067

image-20230801151355027

对于设备而言,一般会订阅平台下发消息给设备 这个主题。

设备想接收平台下发的消息,就需要订阅平台下发消息给设备 的主题,订阅后,平台下发消息给设备,设备就会收到消息。

如果设备想要知道平台下发的消息,需要订阅上面图片里标注的主题。

以当前设备为例,最终订阅主题的格式如下:
$oc/devices/{device_id}/sys/messages/down

最终的格式:
$oc/devices/65113d05a559fd7cd41435f8_lock1/sys/messages/down

(2)主题发布格式

对于设备来说,主题发布表示向云平台上传数据,将最新的传感器数据,设备状态上传到云平台。

这个操作称为:属性上报。

帮助文档地址:https://support.huaweicloud.com/api-iothub/iot_06_v5_3010.html

image-20230925160939978

根据帮助文档的介绍, 当前设备发布主题,上报属性的格式总结如下:

发布的主题格式:
$oc/devices/{device_id}/sys/properties/report
 
最终的格式:
$oc/devices/65113d05a559fd7cd41435f8_lock1/sys/properties/report
发布主题时,需要上传数据,这个数据格式是JSON格式。

上传的JSON数据格式如下:

{
  "services": [
    {
      "service_id": <填服务ID>,
      "properties": {
        "<填属性名称1>": <填属性值>,
        "<填属性名称2>": <填属性值>,
        ..........
      }
    }
  ]
}
根据JSON格式,一次可以上传多个属性字段。 这个JSON格式里的,服务ID,属性字段名称,属性值类型,在前面创建产品的时候就已经介绍了,不记得可以翻到前面去查看。

根据这个格式,组合一次上传的属性数据:
{"services": [{"service_id": "lock","properties":{"lock":1}}]}

3.6 MQTT三元组

MQTT协议登录需要填用户ID,设备ID,设备密码等信息,就像我们平时登录QQ,微信一样要输入账号密码才能登录。MQTT协议登录的这3个参数,一般称为MQTT三元组。

接下来介绍,华为云平台的MQTT三元组参数如何得到。

(1)MQTT服务器地址

要登录MQTT服务器,首先记得先知道服务器的地址是多少,端口是多少。

帮助文档地址:https://console.huaweicloud.com/iotdm/?region=cn-north-4#/dm-portal/home

MQTT协议的端口支持1883和8883,它们的区别是:8883 是加密端口更加安全。但是单片机上使用比较困难,所以当前的设备是采用1883端口进连接的。

根据上面的域名和端口号,得到下面的IP地址和端口号信息: 如果设备支持填写域名可以直接填域名,不支持就直接填写IP地址。 (IP地址就是域名解析得到的)

华为云的MQTT服务器地址:117.78.5.125
域名:e244e6efb9.st1.iotda-device.cn-north-4.myhuaweicloud.com
华为云的MQTT端口号:1883

注意! 具体要看这里:

image-20230925161111346

(2)生成MQTT三元组

华为云提供了一个在线工具,用来生成MQTT鉴权三元组: https://iot-tool.obs-website.cn-north-4.myhuaweicloud.com/

打开这个工具,填入设备的信息(也就是刚才创建完设备之后保存的信息),点击生成,就可以得到MQTT的登录信息了。

下面是打开的页面:

image-20230801151418248

填入设备的信息: (上面两行就是设备创建完成之后保存得到的)

image-20230925161154059

得到三元组之后,设备端通过MQTT协议登录鉴权的时候,填入参数即可。

ClientId 65113d05a559fd7cd41435f8_lock1_0_0_2023092508
Username 65113d05a559fd7cd41435f8_lock1
Password 1a3e7f486aa551bca7b6ff5c19c29d2006e940ec1f98ab416e10be1288106953

3.7 模拟设备登录测试

经过上面的步骤介绍,已经创建了产品,设备,数据模型,得到MQTT登录信息。 接下来就用MQTT客户端软件模拟真实的设备来登录平台。测试与服务器通信是否正常。

(1)填入登录信息

打开MQTT客户端软件,对号填入相关信息(就是上面的文本介绍)。然后,点击登录,订阅主题,发布主题。

image-20230925161345360

(2)打开网页查看

完成上面的操作之后,打开华为云网页后台,可以看到设备已经在线了。

点击详情页面,可以看到上传的数据。

image-20230925161426088

到此,云平台的部署已经完成,设备已经可以正常上传数据了。

3.8 添加控制命令(云端控制设备)

在产品页面,添加控制命令。

image-20230925162251674

根据自己设备的情况填写需要发送的控制命令。

image-20230925162418179

image-20230925162526196

3.9 云端发送控制命令

在设备页面测试刚才添加的命令。

image-20230925162711916

image-20230925162751673

点击确定之后,在MQTT客户端上就能收到云端下发的控制命令。

说明:当前的MQTT客户端实际就是模拟的真实设备。 真实设备收到控制命令就可以判断,完成锁的开关控制。

image-20230925162830773

这个下发的命令是有反馈。设备端收到之后,可以向服务器反馈状态,这样服务器才能知道刚才的控制命令确实发送成功了。

设备收到信息之后,上传回应给服务器的主题和内容格式:

Topic:$oc/devices/{device_id}/sys/commands/response/request_id={request_id}
数据格式:  
{
    "result_code": 0,
    "response_name": "COMMAND_RESPONSE",
    "paras": {
        "result": "success"
    }
}

云端发送控制命令之后,设备收到的消息如下:

$oc/devices/65113d05a559fd7cd41435f8_lock1/sys/commands/request_id=d49f0bb9-ba87-4c9b-b915-98a1f0fcf689{"paras":{"lock":true},"service_id":"lock","command_name":"锁开关控制"}

其中request_id=d49f0bb9-ba87-4c9b-b915-98a1f0fcf689 就是本次的请求ID。回应的时候需要加上请求ID。服务器才好对应。

以当前设备为例:

发布的主题这样填: $oc/devices/65113d05a559fd7cd41435f8_lock1/sys/commands/response/request_id=ce49181e-7636-4b24-946d-c034ca083c1c
    
发布的内容这样填:
{"result_code":0,"response_name":"COMMAND_RESPONSE","paras":{"result":"success"}}

在MQTT软件里这样填:

image-20230925163506474

MQTT回应之后,可以在云端的发送命令界面,看到右上角有弹窗提示,命令下发成功。

image-20230925163208848

四、微信小程序开发

如果要开发自己的上位机,远程控制设备。

一般有两种需求。

(1)获取设备的状态。

(2)给设备发送控制指令。

那么下面就介绍这两种接口如何使用,就是标准的HTTPS接口。

4.1 获取设备影子数据(API接口)

帮助文档:https://support.huaweicloud.com/api-iothub/iot_06_v5_0079.html

设备影子介绍:

设备影子是一个用于存储和检索设备当前状态信息的JSON文档。
每个设备有且只有一个设备影子,由设备ID唯一标识
设备影子仅保存最近一次设备的上报数据和预期数据
无论该设备是否在线,都可以通过该影子获取和设置设备的属性

简单来说:设备影子就是保存,设备最新上传的一次数据。

我们设计的软件里,如果想要获取设备的最新状态信息,就采用设备影子接口。

如果对接口不熟悉,可以先进行在线调试:https://apiexplorer.developer.huaweicloud.com/apiexplorer/doc?product=IoTDA&api=ShowDeviceShadow

在线调试接口,可以请求影子接口,了解请求,与返回的数据格式。

(调试的时候,可以选择只填必填项)

image-20230925161642973

设备影子接口返回的数据如下:

{
 "device_id": "65113d05a559fd7cd41435f8_lock1",
 "shadow": [
  {
   "service_id": "lock",
   "desired": {
    "properties": null,
    "event_time": null
   },
   "reported": {
    "properties": {
     "lock": 1
    },
    "event_time": "20230925T081357Z"
   },
   "version": 0
  }
 ]
}

4.2 控制命令发送(API接口)

帮助文档:https://support.huaweicloud.com/api-iothub/iot_06_v5_0038.html

image-20230925164327371

在线调试地址:https://console.huaweicloud.com/apiexplorer/#/openapi/IoTDA/debug?api=CreateCommand

填写需要发送给设备的消息。

image-20230925164546678

{"lock":1}

设备端可以收到下发的命令

image-20230925164631704

设备端收到下发命令之后,也可以向服务器反馈状态,反馈的方式在前面的3.9小节已经介绍过了。

4.3 创建IAM账户

创建一个IAM账户,因为接下来开发上位机,需要使用云平台的API接口,这些接口都需要token进行鉴权。简单来说,就是身份的认证。 调用接口获取Token时,就需要填写IAM账号信息。所以,接下来演示一下过程。

地址: https://console.huaweicloud.com/iam/?region=cn-north-4#/iam/users

获取Token时,除了AIM账号外,还需要项目凭证:

9254586ba43841e6bf4995dc24023d6b

image-20230925165102886

鼠标点击自己昵称,点击统一身份认证。

点击左上角创建用户

image-20230313175315289

image-20221207161209880

image-20221207161308917

image-20221207161327200

创建成功:

image-20221212174359962

image-20221212174412097

用户创建成功。

image-20230925165154324

4.4 TOKEN动态获取

调用服务器的API接口都需要填写token,下面介绍如何通过账号信息动态获取token(正常的token时效性是24小时)。

帮助文档:https://support.huaweicloud.com/api-iam/iam_30_0001.html#section4

image-20230925170016192

请求地址:

POST https://iam.<服务ID>.myhuaweicloud.com/v3/auth/tokens

请求参数:

{
    "auth": {
        "identity": {
            "methods": [
                "password"
            ],
            "password": {
                "user": {
                    "domain": {
                        "name": "IAMDomain"        //IAM用户所属帐号名(华为云的主账号)
                    },
                    "name": "IAMUser",             //IAM用户名(刚才创建的)
                    "password": "IAMPassword"      //IAM用户密码(刚才创建的)
                }
            }
        },
        "scope": {
            "project": {
                "name": "cn-north-1"               //服务ID(创建产品时创建的)
            }
        }
    }
}

Qt编写的示例代码:

/*
功能: 获取token
*/
void Widget::GetToken()
{
    QString requestUrl;
    QNetworkRequest request;

    //设置请求地址
    QUrl url;

    //获取token请求地址
    requestUrl = QString("https://iam.%1.myhuaweicloud.com/v3/auth/tokens")
                 .arg(SERVER_ID);

    //自己创建的TCP服务器,测试用
    //requestUrl="http://10.0.0.6:8080";

    //设置数据提交格式
    request.setHeader(QNetworkRequest::ContentTypeHeader, QVariant("application/json;charset=UTF-8"));

    //构造请求
    url.setUrl(requestUrl);

    request.setUrl(url);

    QString text =QString("{\"auth\":{\"identity\":{\"methods\":[\"password\"],\"password\":"
    "{\"user\":{\"domain\": {"
    "\"name\":\"%1\"},\"name\": \"%2\",\"password\": \"%3\"}}},"
    "\"scope\":{\"project\":{\"name\":\"%4\"}}}}")
            .arg(MAIN_USER)
            .arg(IAM_USER)
            .arg(IAM_PASSWORD)
            .arg(SERVER_ID);

    //发送请求
    manager->post(request, text.toUtf8());
}

4.5 规则转发

为了能让自己的私有服务器收到设备的状态变化,可以在华为云物联网服务器上创建转发规则。

(1)创建规则

image-20230925170635632

填充规则转发的信息:

参数名参数说明
规则名称自定义,如: led_obs。
规则描述自定义,如数据转发至OBS服务。
数据来源选择“设备属性”。
触发事件自动匹配“设备属性上报”。
资源空间和上一步创建的产品所在的资源空间保持一致。

(2)创建属性:设备上报消息时触动转发规则

image-20230925170830858

image-20230925170914880

image-20230925170929848

(3)添加转发规则

可以根据自己的需求设置目的地址。

image-20230925170959685

如果自己有私有服务器,创建转发规则即可。

image-20230925171126233

五、硬件端代码开发

5.1 E29 Cat-1模块

全网通E29 Cat-1 支持多种协议基于4G网络,具有速度快,延迟低等特点,Cat-1相比于4G有具有更低的功耗。
    
全网通E29Cat-1模块的特点:
1.TAS-E29 Cat-1全网通芯片
2.4G全网通Cat-1芯片
3.支持TCP/UDP/MQTT/HTTP等多种协议
4.基于4G网络,网络速度快
5.功耗比4G芯片低40%
6.支持标准AT指令、短信透传
7.中英文短信透传/配置
8.支持LBS基站定位
9.modbus轮询技术、自定义心跳包、注册包。

5.2 4G模块与MQTT服务器交互常用的AT指令

AT:检查模块是否正常工作并响应AT指令。

AT+CGATT:激活或取消激活GPRS附着。在使用EC20模块进行网络通信之前,需要先激活GPRS附着。

AT+CIPSHUT:关闭移动场景TCP/IP连接。使用该指令可以关闭之前建立的TCP连接。

AT+CIPSTART:建立TCP或UDP连接到远程服务器。对于MQTT协议,首先需要使用AT+CIPSTART建立到MQTT服务器的TCP连接。

AT+CIPSEND:发送数据到TCP或UDP连接。对于MQTT协议,您可以使用该指令发送MQTT数据包,例如订阅主题、发布消息等。

AT+CIPCLOSE:关闭TCP或UDP连接。

AT+CMQTTUSR:设置MQTT连接的用户名。

AT+CMQTTPASS:设置MQTT连接的密码。

AT+CMQTTACCQ:设置MQTT连接的客户端ID。

AT+CMQTTCONN:连接到MQTT服务器。在设置好MQTT连接参数后,使用该指令进行连接。

AT+CMQTTPUB:发布MQTT消息。使用该指令发布消息到指定的主题。

AT+CMQTTSUB:订阅MQTT主题。使用该指令订阅感兴趣的主题,当有消息发布到该主题时,EC20模块会收到相应的回调。

AT+CMQTTUNS:取消订阅MQTT主题。使用该指令取消已经订阅的主题。

5.3 MQTT连接主要代码

核心代码模板:

// 引入必要的库
#include <reg51.h>
#include <string.h>

// 定义串口通信相关的寄存器地址
#define UART_BAUD_RATE_H 0xFD // 波特率高位
#define UART_BAUD_RATE_L 0xFE // 波特率低位
#define UART_CONTROL_REG 0x98 // 控制寄存器地址

// 定义MQTT服务器信息
#define MQTT_SERVER "mqtt.example.com" // MQTT服务器地址
#define MQTT_PORT "1883" // MQTT端口号

// 定义设备ID和认证信息
#define DEVICE_ID "your_device_id" // 设备ID
#define USERNAME "your_username" // 用户名
#define PASSWORD "your_password" // 密码

// 定义订阅和发布的主题
#define SUBSCRIBE_TOPIC "your_subscribe_topic" // 订阅主题
#define PUBLISH_TOPIC "your_publish_topic" // 发布主题

// 定义MQTT协议相关指令
#define CONNECT_CMD "AT+CMQTTCONN=\"" MQTT_SERVER "\",\"" MQTT_PORT "\",\"" DEVICE_ID "\",\"" USERNAME "\",\"" PASSWORD "\"\r\n" // 连接指令
#define SUBSCRIBE_CMD "AT+CMQTTSUB=\"" SUBSCRIBE_TOPIC "\",1\r\n" // 订阅指令
#define PUBLISH_CMD "AT+CMQTTPUB=\"" PUBLISH_TOPIC "\",0,0,0,\"your_message\"\r\n" // 发布指令

// 定义串口发送函数
void UART_SendString(char *str) {
    while (*str) {
        SBUF = *str;
        while (!TI);
        TI = 0;
        str++;
    }
}

// 主函数
void main() {
    // 配置串口通信波特率为9600bps
    PCON = 0x00;
    TMOD = 0x20;
    TH1 = UART_BAUD_RATE_H;
    TL1 = UART_BAUD_RATE_L;
    TR1 = 1;
    SCON = UART_CONTROL_REG;

    // 连接MQTT服务器
    UART_SendString(CONNECT_CMD);

    // 等待连接成功的回复
    // 编写等待回复的代码

    // 订阅主题
    UART_SendString(SUBSCRIBE_CMD);

    // 等待订阅成功的回复
    // 编写等待回复的代码

    // 设备登录成功,进入循环
    while (1) {
        // 定时上报数据,可以根据需求设置时间间隔
        // 这里使用延时函数模拟时间间隔
        delay(5000); 

        // 上报数据到指定主题
        UART_SendString(PUBLISH_CMD);
        
        // 等待上报完成的回复
        // 编写等待回复的代码
    }
}

4.4 锁控制代码

// 引入必要的库
#include <reg51.h>
#include <string.h>

// 定义串口通信相关的寄存器地址
#define UART_BAUD_RATE_H 0xFD // 波特率高位
#define UART_BAUD_RATE_L 0xFE // 波特率低位
#define UART_CONTROL_REG 0x98 // 控制寄存器地址

// 定义MQTT服务器信息
#define MQTT_SERVER "mqtt.example.com" // MQTT服务器地址
#define MQTT_PORT "1883" // MQTT端口号

// 定义设备ID和认证信息
#define DEVICE_ID "your_device_id" // 设备ID
#define USERNAME "your_username" // 用户名
#define PASSWORD "your_password" // 密码

// 定义订阅主题
#define SUBSCRIBE_TOPIC "your_subscribe_topic" // 订阅主题

// 定义电磁锁的IO口连接图纸
#define LOCK_1 P2_0 // 电磁锁1的控制IO口
#define LOCK_2 P2_1 // 电磁锁2的控制IO口
#define LOCK_3 P2_2 // 电磁锁3的控制IO口
#define LOCK_4 P2_3 // 电磁锁4的控制IO口
#define LOCK_5 P2_4 // 电磁锁5的控制IO口
#define LOCK_6 P2_5 // 电磁锁6的控制IO口
#define LOCK_7 P2_6 // 电磁锁7的控制IO口
#define LOCK_8 P2_7 // 电磁锁8的控制IO口
#define LOCK_9 P3_0 // 电磁锁9的控制IO口
#define LOCK_10 P3_1 // 电磁锁10的控制IO口
#define LOCK_11 P3_2 // 电磁锁11的控制IO口
#define LOCK_12 P3_3 // 电磁锁12的控制IO口


// 定义MQTT协议相关指令
#define CONNECT_CMD "AT+CMQTTCONN=\"" MQTT_SERVER "\",\"" MQTT_PORT "\",\"" DEVICE_ID "\",\"" USERNAME "\",\"" PASSWORD "\"\r\n" // 连接指令
#define SUBSCRIBE_CMD "AT+CMQTTSUB=\"" SUBSCRIBE_TOPIC "\",1\r\n" // 订阅指令

// 定义串口发送函数
void UART_SendString(char *str) {
    while (*str) {
        SBUF = *str;
        while (!TI);
        TI = 0;
        str++;
    }
}

// 电磁锁控制函数
void controlLock(int lockNumber, int lockState) {
    switch (lockNumber) {
    case 3:
        // 控制电磁锁3
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_3 = 1;
        } else {
            // 关闭电磁锁
            LOCK_3 = 0;
        }
        break;
    case 4:
        // 控制电磁锁4
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_4 = 1;
        } else {
            // 关闭电磁锁
            LOCK_4 = 0;
        }
        break;
    case 5:
        // 控制电磁锁5
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_5 = 1;
        } else {
            // 关闭电磁锁
            LOCK_5 = 0;
        }
        break;
    case 6:
        // 控制电磁锁6
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_6 = 1;
        } else {
            // 关闭电磁锁
            LOCK_6 = 0;
        }
        break;
    case 7:
        // 控制电磁锁7
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_7 = 1;
        } else {
            // 关闭电磁锁
            LOCK_7 = 0;
        }
        break;
    case 8:
        // 控制电磁锁8
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_8 = 1;
        } else {
            // 关闭电磁锁
            LOCK_8 = 0;
        }
        break;
    case 9:
        // 控制电磁锁9
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_9 = 1;
        } else {
            // 关闭电磁锁
            LOCK_9 = 0;
        }
        break;
    case 10:
        // 控制电磁锁10
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_10 = 1;
        } else {
            // 关闭电磁锁
            LOCK_10 = 0;
        }
        break;
    case 11:
        // 控制电磁锁11
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_11 = 1;
        } else {
            // 关闭电磁锁
            LOCK_11 = 0;
        }
        break;
    case 12:
        // 控制电磁锁12
        if (lockState == 1) {
            // 打开电磁锁
            LOCK_12 = 1;
        } else {
            // 关闭电磁锁
            LOCK_12 = 0;
        }
        break;
    default:
        // 锁编号无效
        break;
}

}

// 主函数
void main() {
    // 配置串口通信波特率为9600bps
    PCON = 0x00;
    TMOD = 0x20;
    TH1 = UART_BAUD_RATE_H;
    TL1 = UART_BAUD_RATE_L;
    TR1 = 1;
    SCON = UART_CONTROL_REG;

    // 连接MQTT服务器
    UART_SendString(CONNECT_CMD);

    // 等待连接成功的回复
    // 编写等待回复的代码

    // 订阅主题
    UART_SendString(SUBSCRIBE_CMD);

    // 等待订阅成功的回复
    // 编写等待回复的代码

    // 设备登录成功,进入循环
    while (1) {
        // 接收到MQTT消息后,解析并控制对应的电磁锁
        // 编写接收和解析MQTT消息的代码
        // 设备接收到的消息格式为 "LOCK_X_ON" 或 "LOCK_X_OFF",其中 X 为电磁锁编号

        // 解析消息,并提取电磁锁编号和状态
        int lockNumber; // 电磁锁编号
        int lockState; // 电磁锁状态,0表示关闭,1表示打开
        if (strncmp(receivedMessage, "LOCK_", 5) == 0) {
            lockNumber = receivedMessage[5] - '0'; // 提取锁编号
            if (strstr(receivedMessage, "ON") != NULL) {
                lockState = 1; // 提取锁状态,ON表示打开
            } else if (strstr(receivedMessage, "OFF") != NULL) {
                lockState = 0; // 提取锁状态,OFF表示关闭
            }

            // 控制对应的电磁锁
            controlLock(lockNumber, lockState);
        }
    }
}

六、总结

这款基于单片机设计的智能储物柜是一种应用于游乐场、商场和景区等场所的便捷储物解决方案。用户可以通过微信小程序实时查看储物柜的可用状态,并选择指定柜子进行解锁。

该智能储物柜支持4G联网,采用主控芯片STC12C5A60S2和4G联网模块E29 Cat-1来实现与网络的连接。用户在解锁储物柜时需要支付押金(预付金),随后系统开始计时并按照使用的时间进行收费。储物柜也支持临时解锁和存取物品功能。用户完成使用后,可以通过微信小程序完成订单结算。

储物柜的锁采用电磁锁技术,通过控制相应的IO口来实现开锁或关锁。整体而言,该智能储物柜以单片机为核心,集成了4G通信模块、电磁锁和华为云物联网服务器等技术,提供方便、安全的储物服务。每个储物柜都是一个独立的设备,通过华为云物联网服务器实现数据传输和管理。

基于单片机设计的智能储物柜在不同场所为用户提供便捷的储物解决方案,通过技术的融合和应用,提升用户体验和管理效率。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

矩阵快速幂技巧练习(一)— 经典牛问题

上一篇文章简单介绍了斐波那契数列的矩阵乘法&#xff0c;并做了一个小推广&#xff0c;这篇文章来小试牛刀&#xff0c;做一个经典的练习题。 求斐波那契数列矩阵乘法的方法 题目 第一年农场有一只成熟的母牛A&#xff0c;往后的每年&#xff1a; 每一只成熟的母牛都会生一只…

MySQL面试题 | 09.精选MySQL面试题

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

【Debian】非图形界面Debian10.0.0安装xfce和lxde桌面

一、安装 1. Debian10.0.0安装xfce桌面 sudo apt update sudo apt install xfce4 startxfce4 2. Debian10.0.0安装lxde桌面 sudo apt-get install lxde安装后重启电脑。 二、说明 XFCE、LXDE 和 GNOME 是三个流行的桌面环境&#xff0c;它们都是为类 Unix 操作系统设计…

C语言:编译和链接

目录 一&#xff1a;翻译环境和运行环境 二&#xff1a;翻译环境 2.1预处理&#xff08;预编译&#xff09; 2.2编译 2.2.1 词法分析&#xff1a; 2.2.2语法分析 2.2.3语义分析 2.3 汇编 三&#xff1a;运行环境 一&#xff1a;翻译环境和运行环境 在ANSI C的任何一种…

微信小程序------WXML模板语法之条件渲染和列表渲染

目录 前言 一、条件渲染 1.wx:if 2. 结合 使用 wx:if 3. hidden 4. wx:if 与 hidden 的对比 二、列表渲染 1. wx:for 2. 手动指定索引和当前项的变量名* 3. wx:key 的使用 前言 上一期我们讲解wxml模版语法中的数据绑定和事件绑定&#xff08;上一期链接&#xff1a;…

MATLAB - 使用运动学 DH 参数构建机械臂

系列文章目录 前言 一、 使用 Puma560 机械手机器人的 Denavit-Hartenberg (DH) 参数&#xff0c;逐步建立刚体树形机器人模型。在连接每个关节时&#xff0c;指定其相对 DH 参数。可视化机器人坐标系&#xff0c;并与最终模型进行交互。 DH 参数定义了每个刚体通过关节与其父…

Go-gin-example 第二部分 jwt验证

文章目录 使用 JWT 进行身份校验jwt知识点补充认识JWTTOKEN是什么jwt的使用场景jwt的组成headerpayloadsignature 下载依赖包编写 jwt 工具包jwt中间件编写如何获取token 编写获取token的Apimodels逻辑编写路由逻辑编写修改路由逻辑 验证token将中间件接入Gin功能验证模块 续接…

gitlab 命令执行漏洞(CVE-2022-2992)

1.漏洞影响版本 GitLab CE/EE 中的一个漏洞影响从 11.10 开始到 15.1.6 之前的所有版本、从 15.2 开始到 15.2.4 之前的所有版本、从 15.3 开始到 15.3.2 之前的所有版本。允许经过身份验证的用户通过从 GitHub API 端点导入实现远程代码执行。 查看 gitlab 版本。(登录后才能…

【目标检测】YOLOv7算法实现(一):模型搭建

本系列文章记录本人硕士阶段YOLO系列目标检测算法自学及其代码实现的过程。其中算法具体实现借鉴于ultralytics YOLO源码Github&#xff0c;删减了源码中部分内容&#xff0c;满足个人科研需求。   本系列文章在YOLOv5算法实现的基础上&#xff0c;进一步完成YOLOv7算法的实现…

使用STM32Cube库开发USB虚拟串口设备

开发基于STM32Cube库的USB虚拟串口设备需要了解USB通信协议、虚拟串口设备的基本原理以及STM32Cube库的使用。在这篇文章中&#xff0c;我们将介绍如何利用STM32Cube库开发一个USB虚拟串口设备&#xff0c;并提供相应的代码示例。 1. USB虚拟串口设备概述 USB虚拟串口设备是指…

力扣刷题(无重复字符的最长子串)

3. 无重复字符的最长子串https://leetcode.cn/problems/longest-substring-without-repeating-characters/ 给定一个字符串 s &#xff0c;请你找出其中不含有重复字符的 最长子串 的长度。 示例 1: 输入: s "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是…

使用vue快速开发一个带弹窗的Chrome插件

vue-chrome-extension-quickstart 说在前面 &#x1f388;平时我们使用Chrome插件通常都只是用来编写简单的js注入脚本&#xff0c;大家有没有遇到过需要插件在页面上注入一个弹窗呢&#xff1f;比如我们希望可以通过快捷键快速唤起ChatGPT面板或者快速唤起一个翻译面板&#x…

动态规划:01背包问题(一)

本题力扣上没有&#xff0c;是刷的卡码网第46题感兴趣的小伙伴可以去刷一下&#xff0c;是ACM模式。本篇讲解二维dp数组来解决01背包问题&#xff0c;下篇博客将用一维dp数组来解决01背包问题。 题目&#xff1a; 46. 携带研究材料 时间限制&#xff1a;5.000S 空间限制&…

Spark---RDD持久化

文章目录 1.RDD持久化1.1 RDD Cache 缓存1.2 RDD CheckPoint 检查点1.3 缓存和检查点区别 2.RDD分区器2.1 Hash 分区&#xff1a;2.2 Range 分区&#xff1a;2.3 用户自定义分区 1.RDD持久化 在Spark中&#xff0c;持久化是将RDD存储在内存中&#xff0c;以便在多次计算之间重…

HDFS WebHDFS 读写文件分析及HTTP Chunk Transfer Encoding相关问题探究

文章目录 前言需要回答的首要问题DataNode端基于Netty的WebHDFS Service的实现基于重定向的文件写入流程写入一个大文件时WebHDFS和Hadoop Native的块分布差异 基于重定向的数据读取流程尝试读取一个小文件尝试读取一个大文件 读写过程中的Chunk Transfer-Encoding支持写文件使…

快慢指针-Floyd判圈算法

对于环形链表是否存在环的做法&#xff0c;普通算法可以通过额外Hash数组来存储链表元素&#xff0c;直到Hash数组中出现重复元素。时间复杂度O(n)&#xff0c;空间复杂度O(n) Floyd判圈算法通过利用快慢指针的移动来实现&#xff0c;时间复杂度O&#xff08;n&#xff09;&am…

Elasticsearch:聊天机器人教程(二)

这是继上一篇文章 “Elasticsearch&#xff1a;聊天机器人教程&#xff08;一&#xff09;”的续篇。本教程的这一部分讨论聊天机器人实现中最有趣的方面&#xff0c;以帮助你理解它并对其进行自定义。 数据摄入 在此应用程序中&#xff0c;所有示例文档的摄取都是通过 flask …

搭建知识付费小程序平台:如何避免被坑,选择最佳方案?

随着知识经济的兴起&#xff0c;知识付费已经成为一种趋势。越来越多的人开始将自己的知识和技能进行变现&#xff0c;而知识付费小程序平台则成为了一个重要的渠道。然而&#xff0c;市面上的知识付费小程序平台琳琅满目&#xff0c;其中不乏一些不良平台&#xff0c;让老实人…

git提交报错:remote: Please remove the file from history and try again.

1. 报错信息 remote: error: File: fba7046b22fd74b77425aa3e4eae0ea992d44998 500.28 MB, exceeds 100.00 MB. remote: Please remove the file from history and try again. git rev-list --objects --all | grep fba7046b22fd74b77425aa3e4eae0ea992d44998 2. 分析原因 e…

单例模式实现最好的方式即枚举实现

单例类作为23种设计模式当中最常用的设计模式&#xff0c;实现方式有很多种&#xff0c;比较流行的是DCL(DoubleCheckLock)双重检查的实现&#xff0c;线程安全&#xff0c;又比较好&#xff0c;除了存在序列化的问题之外&#xff0c;还算不错&#xff0c;如果对DCL模式还不熟悉…