1. 前言
我国独生子女,以及人口老龄化等问题,正逐渐成为一个重大的社会问题,老年人机体能力的下降,摔倒引起的安全和危害愈来愈突出,国家和社会越来越关注老年人的健康和安全,开发一个能够实时检测出老年人是否摔倒,并且能及时告知监护人的摔倒检测以及报警系统具有重要的现实意义。本系统包括检测摔倒模块、GPS定位模块和通信模块三部分,通过检测老年人日常状态,可以得知老年人的状态,如果监测到老年人摔倒了,此时会通过网络把检测结果上传到物联网云平台,获得老年人摔倒地点的GPS定位,并且通过GPRS通讯发短信给预设的监护人。
2. 设计需求
(1)采用STM32单片机作为主控芯片,配合其他模块完成功能设计
(2)通信模块采用SIM800C,支持上传采集的GPS经纬度数据到云端服务器,云端采用华为云物联网平台。
(3)老人摔倒检测采用MPU6050陀螺仪检测,当检测到老人摔倒之后,会通过SIM800C发送短信到紧急联系人,设备上的蜂鸣器会发出警报声,周围行人听到也可以进行帮助;并且会将GPS数据上传到云端,通过地图显示老人的位置,家人通过短信知道老人摔倒后,通过云端地图显示的位置,可以快速赶到老人身边,或者报警求助,报告位置。
(4)老人摔倒后,如果自己能行动,没有大问题,可以自己按下设备上的按键取消蜂鸣器报警,并且通过SIM800C向家人发送一条短信,报平安。
3. 设计的实物效果
为了快速验证方案的可行性,这里采用现成的模块采用杜邦线连接完成整个预想的功能设计。
下面就是硬件连接好之后的效果图,选用的硬件型号在第4章节已经全部贴出来了;为了方便户外测试,这里的供电电源采用了充电宝,也可以采用电池盒供电。



可以设置电子围栏,坐标超出之后进行提示。



4. 硬件选型
主控芯片采用STM32RCT6,通信模块采用SIM800C,GPS采集使用ATGM336北斗BDS+GPS双模模块,老人摔倒检测模块采用MPU6050陀螺仪。
这些都是采用现成的成品模块,都是在淘宝上买的,下面都贴出了模块的型号,模块的实物截图,如果自己想做一个,可以在淘宝上找到一样的模块型号购买。
4.1 SIM800C
SIM800C模块是一款高性能高性价比工业级的GSM/GPRS模块。本模块采用SIMCOM公司的工业级四频850/900/ 1800/1900MHz SM800芯片,可以低功耗实现语音、SMS、数据和传真信息的传输。

模块特点:
1、支持极限DC5V-18V宽电压输入
2、有电源使能开关引脚EN
3、支持锂电池供电接口VBAT3.5-4.5V
4、输入支持移动和联通手机卡Micro SIM卡
5、送51/STM32/ARDUINO驱动例程
1、DC 5V-18V电源输入,推荐使用DC 9V
2、电源开始使能引脚默认使能
3、电源地
4、GSM模块的TXD引脚接其它模块的RXD
5、GSM模块的RXD引脚接其它模块的TXD
6、数据终端准备
7、内核音频输出引脚
8、内核音频输出引脚
9、锂电池输入引脚,DC 3.5 - 4.5V
10、电源地

11、启动引脚和GND短路可实现开机自启动
12、电源地
13、RTC外置电池引脚
14、内核振铃提示引脚
15、内合音频输入引脚
16、内核音频输入引脚
加粗的引脚一般都用到。
建议使用V_IN单独供电DC5-18V输入(推荐使用9V),或者VBAT供电锂电池两种供电方式这两种供电方式最稳定。如果只是简单调试,也可使用USB-TTL或者开发板的5V直接给模块供电。不过一般电脑或者开发板的功率有限,可能会不稳定。请根据具体情况自己取舍选择合适电源。
总结:
模块本身支持自适应波特率,可以自动根据发送过去的指令计算对应的波特率,一般使用115200即可。

模块调试总结:
(1)供电电压5V也可以,采用电脑USB供电(直接插电脑USB口)。正常供电之后,模块上有电源指示灯。
(2)SIM800C的TX脚接单片机的RX脚
(3)SIM800C的RX脚接单片机的TX脚
(4)SIM800C的第11个引脚(PWK)和12个引脚(GND)短接接在一起,才可以开机。

电源正常后,右上角有一个黄色的电源灯。

通过串口发送AT指令过去测试模块效果。

4.2 STM32F103C8T6开发板

3.3 GPS模块
GPS模块正常定位后,模块上的LED灯会按照1秒钟闪烁一次。
返回的字段里GNRMC表示当前定位的GPS经纬度,解析代码只需要解析GNRMC字段。
第一次启动GPS模块,定位差不多要几分钟时间,定位成功后,第二次启动定位就很快,最好是在室外,室内信号差,定位时间更久。
$GNGGA,114955.000,2842.4158,N,11549.5439,E,1,05,3.8,54.8,M,0.0,M,,*4F
$GNGLL,2842.4158,N,11549.5439,E,114955.000,A,A*4D
$GPGSA,A,3,10,31,18,,,,,,,,,,5.7,3.8,4.2*37
$BDGSA,A,3,07,10,,,,,,,,,,,5.7,3.8,4.2*2A
$GPGSV,3,1,10,10,49,184,42,12,16,039,,14,54,341,,18,22,165,23*7B
$GPGSV,3,2,10,22,11,318,,25,51,055,,26,24,205,,29,13,110,*7C
$GPGSV,3,3,10,31,50,287,36,32,66,018,*7F
$BDGSV,1,1,04,03,,,07,05,,,29,07,79,246,33,10,52,232,19*62
$GNRMC,114955.000,A,2842.4158,N,11549.5439,E,0.00,44.25,061117,,,A*4D
$GNVTG,44.25,T,,M,0.00,N,0.00,K,A*14
$GNZDA,114955.000,06,11,2017,00,00*47
$GPTXT,01,01,01,ANTENNA OK*35
- 1.
- 2.
- 3.
- 4.
- 5.
- 6.
- 7.
- 8.
- 9.
- 10.
- 11.
- 12.

4.4 MPU6050陀螺仪
陀螺仪选择的是正点原子的模块,比较稳定,质量较好。

4.5 蜂鸣器
蜂鸣器选择的高电平触发。

5. 创建云端物联网服务器
为了方便查看老人摔倒之后的位置,需要通过SIM800C将设备采集的GPS数据上传到云平台服务器保存,就算老人没有跌倒,也可以实时关注老人的位置,在地图上绘制出轨迹线路,方便家人随时联系,了解老人的情况。
这里物联网的平台选择是华为云物联网平台,目前是免费使用的,在云端创建产品等信息后,设备再通过MQTT协议连接云平台上传GPS数据。目前华为云的拖拽试网页开发页面已经下架,目前要开发对应的上位机,可以采用最近主推的低代码开发平台或者自己通过云平台的应用侧开发接口自己开发上位机,我这里是自己开发的上位机,通过QT编写的上位机APP,支持windows、Android、Linux等多个平台运行,跨平台使用还是比较方便。
下面接着就介绍如何登陆官网创建产品、设备、完成云端的产品部署。
5.1 创建产品
官网地址: https://www.huaweicloud.com/product/iothub.html
打开官网后没有华为云账号需要先注册账号,这些步骤就不多说了,接下来就直接介绍如何创建产品、设备、配置属性、完成数据上传交互的流程。

点击免费使用进去页面。

点击左边产品选项,点击右上角创建产品按钮,弹出参数填充对话框。

根据自己的设备情况填入信息之后保存。

产品创建成功,点击查看详细信息。


5.2 创建模型文件
在现在的详情页面往下翻,可以看到模型创建的选项。
点击自定义模型选项,创建模型。
这里的模型就是设备上传的数据属性。

添加服务ID。

点击创建属性,这里选择JSON类型的数据,上传的GPS有经纬度两个数据,方便保存。

创建成功。

5.3 创建设备
产品是一个大框架的模型,下面可以创建很多具体的设备,目前我这里只有一个硬件设备,就创建一个设备就行了。设备可以手动创建,也支持自动创建,就像现在市面上的智能设备产品,拿到设备后,扫描设备上二维码再通手机APP就可以完成产品的创建,设备的添加。 目前我这里设备就只有一个,而且还要演示整个流程,就在网页上完成整个设备的创建。
链接地址: https://console.huaweicloud.com/iotdm/?region=cn-north-4#/dm-portal/device/all-device
点击左边的设备选项,再点击右上角的注册设备。

填充好信息之后,点击确定。

创建后保存设备的数据。

创建成功,目前设备处于未激活状态。

5.4 获取MQTT登录参数
目前产品、设备创建好之后就需要通过设备连接上来上传数据,要完成这个步骤,还需要知道一些前提的流程。
【A】华为云服务器IP地址、域名、端口号
【B】主题订阅的格式、主题发布的格式
【C】MQTT协议登录的三元组信息
充分了解了这3个信息之后就可以编写设备端代码了。下面就详细介绍这些信息怎么得到。
【1】华为云的服务器地址信息
在这里查看: https://console.huaweicloud.com/iotdm/?region=cn-north-4#/dm-portal/instance/detail?id=8fe87243-d97d-4c1e-bb34-186a60ca2d14&type=public
【2】主题订阅的格式、主题发布的格式
主题订阅上报的格式在产品的详情页面可以看到。

主题发布官方的详细介绍在这里:
https://support.huaweicloud.com/devg-iothub/iot_01_2127.html#ZH-CN_TOPIC_0240834853__zh-cn_topic_0251997880_li365284516112

主题上报的属性格式说明文档地址:
https://support.huaweicloud.com/api-iothub/iot_06_v5_3010.html

根据当前设备的信息总结,得到的信息如下:
【3】MQTT协议登录的三元组信息
华为云提供了MQTT协议参数的生成工具,非常方便,根据提示填入参数一键生成三元组。
MQTT设备登陆密匙生成地址: https://iot-tool.obs-website.cn-north-4.myhuaweicloud.com/

得到的三元组如下:
5.5 采用MQTT客户工具登录测试
为了方便验证服务器的配置以及主题、属性这些是否OK,可以先使用MQTT客户端模拟真实设备登录测试。下面这个MQTT工具是我自己开发的,为了方便测试对接物联网平台,使用QT写了这么一个工具软件。
工具软件的名称: MQTT客户端_v2.5(协议3.1.1).exe
下面的文章的附件里我会上传一份。
在软件左边根据提示填入对应的参数,依次点击登录,订阅主题,发布主题即可。

这时打开网页可以看到设备已经在线了。

在设备影子页面上可以看到上传的数据内容。

启动消息跟踪,可以了解通信的过程。

6. STM32硬件设备端程序设计
在第5章完成了物联网云平台的构建,接下来的第6章节,就编写STM32设备端代码。
STM32设备端开发环境采用keil5进行开发,编程风格采用寄存器风格形式,不管使用库函数,还是寄存器,还是HAL库,本身都一样,没有太大区别,我编写STM32代码习惯了寄存器开发,主要是寄存器的代码比较简洁,工程文件精简。
关于keil5软件的下载流程、安装流程、基本使用办法这里就不在详细介绍,相应看这篇文章的道友
应该这些会这些基操,这里主要是以项目为导向,介绍比较核心的知识点。
6.1 硬件接线
下面是介绍使用的硬件模块与STM32开发板之间的硬件连线。
这是通过杜邦线接好模块后的效果图:

6.2 keil工程截图


6.3 原理图
下面是绘制的原理图。

6.4 MQTT协议实现代码以及MQTT参数
SIM800C本身没有内置MQTT协议指令,只有TCP通信的指令,需要自己封装MQTT协议,然后通过TCP通信的相关指令完成云端服务器连接,实现数据交互。
下面这份代码是MQTT协议的参数定义,程序里为了方便修改,采用宏定义方式赋值这些参数。
这是封装的几个MQTT协议核心函数:
6.4 MPU6050.c代码
这是MPU6050陀螺仪的核心驱动代码,方便检测老人的姿态,判断是否摔倒。
#include "mpu6050.h"
#include "sys.h"
#include "delay.h"
#include <stdio.h>
void MPU6050_IIC_Delay(void)
{
DelayUs(2);
}
void MPU6050_IIC_Init(void)
{
RCC->APB2ENR|=1<<3;
GPIOB->CRL&=0X00FFFFFF;
GPIOB->CRL|=0X33000000;
GPIOB->ODR|=3<<6;
}
void MPU6050_IIC_Start(void)
{
MPU6050_SDA_OUT();
MPU6050_IIC_SDA=1;
MPU6050_IIC_SCL=1;
MPU6050_IIC_Delay();
MPU6050_IIC_SDA=0;
MPU6050_IIC_Delay();
MPU6050_IIC_SCL=0;
}
void MPU6050_IIC_Stop(void)
{
MPU6050_SDA_OUT();
MPU6050_IIC_SCL=0;
MPU6050_IIC_SDA=0;
MPU6050_IIC_Delay();
MPU6050_IIC_SCL=1;
MPU6050_IIC_SDA=1;
MPU6050_IIC_Delay();
}
u8 MPU6050_IIC_Wait_Ack(void)
{
u8 ucErrTime=0;
MPU6050_SDA_IN();
MPU6050_IIC_SDA=1;MPU6050_IIC_Delay();
MPU6050_IIC_SCL=1;MPU6050_IIC_Delay();
while(MPU6050_READ_SDA)
{
ucErrTime++;
if(ucErrTime>250)
{
MPU6050_IIC_Stop();
return 1;
}
}
MPU6050_IIC_SCL=0;
return 0;
}
void MPU6050_IIC_Ack(void)
{
MPU6050_IIC_SCL=0;
MPU6050_SDA_OUT();
MPU6050_IIC_SDA=0;
MPU6050_IIC_Delay();
MPU6050_IIC_SCL=1;
MPU6050_IIC_Delay();
MPU6050_IIC_SCL=0;
}
void MPU6050_IIC_NAck(void)
{
MPU6050_IIC_SCL=0;
MPU6050_SDA_OUT();
MPU6050_IIC_SDA=1;
MPU6050_IIC_Delay();
MPU6050_IIC_SCL=1;
MPU6050_IIC_Delay();
MPU6050_IIC_SCL=0;
}
void MPU6050_IIC_Send_Byte(u8 txd)
{
u8 t;
MPU6050_SDA_OUT();
MPU6050_IIC_SCL=0;
for(t=0;t<8;t++)
{
MPU6050_IIC_SDA=(txd&0x80)>>7;
txd<<=1;
MPU6050_IIC_SCL=1;
MPU6050_IIC_Delay();
MPU6050_IIC_SCL=0;
MPU6050_IIC_Delay();
}
}
u8 MPU6050_IIC_Read_Byte(unsigned char ack)
{
unsigned char i,receive=0;
MPU6050_SDA_IN();
for(i=0;i<8;i++ )
{
MPU6050_IIC_SCL=0;
MPU6050_IIC_Delay();
MPU6050_IIC_SCL=1;
receive<<=1;
if(MPU6050_READ_SDA)receive++;
MPU6050_IIC_Delay();
}
if(!ack)
MPU6050_IIC_NAck();
else
MPU6050_IIC_Ack();
return receive;
}
#define MPU6050_AD0_CTRL PAout(15)
u8 MPU6050_Init(void)
{
u8 res;
RCC->APB2ENR|=1<<2;
GPIOA->CRH&=0X0FFFFFFF;
GPIOA->CRH|=0X30000000;
JTAG_Set(SWD_ENABLE);
MPU6050_AD0_CTRL=0;
MPU6050_IIC_Init();
MPU6050_Write_Byte(MPU_PWR_MGMT1_REG,0X80);
DelayMs(100);
MPU6050_Write_Byte(MPU_PWR_MGMT1_REG,0X00);
MPU6050_Set_Gyro_Fsr(3);
MPU6050_Set_Accel_Fsr(0);
MPU6050_Set_Rate(50);
MPU6050_Write_Byte(MPU_INT_EN_REG,0X00);
MPU6050_Write_Byte(MPU_USER_CTRL_REG,0X00);
MPU6050_Write_Byte(MPU_FIFO_EN_REG,0X00);
MPU6050_Write_Byte(MPU_INTBP_CFG_REG,0X80);
res=MPU6050_Read_Byte(MPU_DEVICE_ID_REG);
if(res==MPU6050_ADDR)
{
MPU6050_Write_Byte(MPU_PWR_MGMT1_REG,0X01);
MPU6050_Write_Byte(MPU_PWR_MGMT2_REG,0X00);
MPU6050_Set_Rate(50);
}else return 1;
return 0;
}
u8 MPU6050_Set_Gyro_Fsr(u8 fsr)
{
return MPU6050_Write_Byte(MPU_GYRO_CFG_REG,fsr<<3);
}
u8 MPU6050_Set_Accel_Fsr(u8 fsr)
{
return MPU6050_Write_Byte(MPU_ACCEL_CFG_REG,fsr<<3);
}
u8 MPU6050_Set_LPF(u16 lpf)
{
u8 data=0;
if(lpf>=188)data=1;
else if(lpf>=98)data=2;
else if(lpf>=42)data=3;
else if(lpf>=20)data=4;
else if(lpf>=10)data=5;
else data=6;
return MPU6050_Write_Byte(MPU_CFG_REG,data);
}
u8 MPU6050_Set_Rate(u16 rate)
{
u8 data;
if(rate>1000)rate=1000;
if(rate<4)rate=4;
data=1000/rate-1;
data=MPU6050_Write_Byte(MPU_SAMPLE_RATE_REG,data);
return MPU6050_Set_LPF(rate/2);
}
short MPU6050_Get_Temperature(void)
{
u8 buf[2];
short raw;
float temp;
MPU6050_Read_Len(MPU6050_ADDR,MPU_TEMP_OUTH_REG,2,buf);
raw=((u16)buf[0]<<8)|buf[1];
temp=36.53+((double)raw)/340;
return temp*100;;
}
u8 MPU6050_Get_Gyroscope(short *gx,short *gy,short *gz)
{
u8 buf[6],res;
res=MPU6050_Read_Len(MPU6050_ADDR,MPU_GYRO_XOUTH_REG,6,buf);
if(res==0)
{
*gx=((u16)buf[0]<<8)|buf[1];
*gy=((u16)buf[2]<<8)|buf[3];
*gz=((u16)buf[4]<<8)|buf[5];
}
return res;;
}
u8 MPU6050_Get_Accelerometer(short *ax,short *ay,short *az)
{
u8 buf[6],res;
res=MPU6050_Read_Len(MPU6050_ADDR,MPU_ACCEL_XOUTH_REG,6,buf);
if(res==0)
{
*ax=((u16)buf[0]<<8)|buf[1];
*ay=((u16)buf[2]<<8)|buf[3];
*az=((u16)buf[4]<<8)|buf[5];
}
return res;;
}
void MPU6050_Get_Data(short *ax,short *ay,short *az)
{
short ax_buff[10]={0};
short ay_buff[10]={0};
short az_buff[10]={0};
int i=0,j=0;
short tmp;
u32 data=0;
for(i=0;i<10;i++)
{
MPU6050_Get_Accelerometer(&ax_buff[i],&ay_buff[i],&az_buff[i]);
}
for(i=0;i<10-1;i++)
{
for(j=0;j<10-1-i;j++)
{
if(ax_buff[j]>ax_buff[j+1])
{
tmp=ax_buff[j];
ax_buff[j]=ax_buff[j+1];
ax_buff[j+1]=tmp;
}
}
}
for(i=1;i<9;i++)
{
data+=ax_buff[i];
}
*ax=data/8;
for(i=0;i<10-1;i++)
{
for(j=0;j<10-1-i;j++)
{
if(ay_buff[j]>ay_buff[j+1])
{
tmp=ay_buff[j];
ay_buff[j]=ay_buff[j+1];
ay_buff[j+1]=tmp;
}
}
}
for(i=1;i<9;i++)
{
data+=ay_buff[i];
}
*ay=data/8;
for(i=0;i<10-1;i++)
{
for(j=0;j<10-1-i;j++)
{
if(az_buff[j]>az_buff[j+1])
{
tmp=az_buff[j];
az_buff[j]=az_buff[j+1];
az_buff[j+1]=tmp;
}
}
}
for(i=1;i<9;i++)
{
data+=az_buff[i];
}
*az=data/8;
}
u8 MPU6050_Write_Len(u8 addr,u8 reg,u8 len,u8 *buf)
{
u8 i;
MPU6050_IIC_Start();
MPU6050_IIC_Send_Byte((addr<<1)|0);
if(MPU6050_IIC_Wait_Ack())
{
MPU6050_IIC_Stop();
printf("等待应答失败\r\n");
return 1;
}
MPU6050_IIC_Send_Byte(reg);
MPU6050_IIC_Wait_Ack();
for(i=0;i<len;i++)
{
MPU6050_IIC_Send_Byte(buf[i]);
if(MPU6050_IIC_Wait_Ack())
{
MPU6050_IIC_Stop();
printf("等待ACK失败\r\n");
return 1;
}
}
MPU6050_IIC_Stop();
return 0;
}
u8 MPU6050_Read_Len(u8 addr,u8 reg,u8 len,u8 *buf)
{
MPU6050_IIC_Start();
MPU6050_IIC_Send_Byte((addr<<1)|0);
if(MPU6050_IIC_Wait_Ack())
{
MPU6050_IIC_Stop();
return 1;
}
MPU6050_IIC_Send_Byte(reg);
MPU6050_IIC_Wait_Ack();
MPU6050_IIC_Start();
MPU6050_IIC_Send_Byte((addr<<1)|1);
MPU6050_IIC_Wait_Ack();
while(len)
{
if(len==1)*buf=MPU6050_IIC_Read_Byte(0);
else *buf=MPU6050_IIC_Read_Byte(1);
len--;
buf++;
}
MPU6050_IIC_Stop();
return 0;
}
u8 MPU6050_Write_Byte(u8 reg,u8 data)
{
MPU6050_IIC_Start();
MPU6050_IIC_Send_Byte((MPU6050_ADDR<<1)|0);
if(MPU6050_IIC_Wait_Ack())
{
MPU6050_IIC_Stop();
return 1;
}
MPU6050_IIC_Send_Byte(reg);
MPU6050_IIC_Wait_Ack();
MPU6050_IIC_Send_Byte(data);
if(MPU6050_IIC_Wait_Ack())
{
MPU6050_IIC_Stop();
return 1;
}
MPU6050_IIC_Stop();
return 0;
}
u8 MPU6050_Read_Byte(u8 reg)
{
u8 res;
MPU6050_IIC_Start();
MPU6050_IIC_Send_Byte((MPU6050_ADDR<<1)|0);
MPU6050_IIC_Wait_Ack();
MPU6050_IIC_Send_Byte(reg);
MPU6050_IIC_Wait_Ack();
MPU6050_IIC_Start();
MPU6050_IIC_Send_Byte((MPU6050_ADDR<<1)|1);
MPU6050_IIC_Wait_Ack();
res=MPU6050_IIC_Read_Byte(0);
MPU6050_IIC_Stop();
return res;
}
- 1.
- 2.
- 3.
- 4.
- 5.
- 6.
- 7.
- 8.
- 9.
- 10.
- 11.
- 12.
- 13.
- 14.
- 15.
- 16.
- 17.
- 18.
- 19.
- 20.
- 21.
- 22.
- 23.
- 24.
- 25.
- 26.
- 27.
- 28.
- 29.
- 30.
- 31.
- 32.
- 33.
- 34.
- 35.
- 36.
- 37.
- 38.
- 39.
- 40.
- 41.
- 42.
- 43.
- 44.
- 45.
- 46.
- 47.
- 48.
- 49.
- 50.
- 51.
- 52.
- 53.
- 54.
- 55.
- 56.
- 57.
- 58.
- 59.
- 60.
- 61.
- 62.
- 63.
- 64.
- 65.
- 66.
- 67.
- 68.
- 69.
- 70.
- 71.
- 72.
- 73.
- 74.
- 75.
- 76.
- 77.
- 78.
- 79.
- 80.
- 81.
- 82.
- 83.
- 84.
- 85.
- 86.
- 87.
- 88.
- 89.
- 90.
- 91.
- 92.
- 93.
- 94.
- 95.
- 96.
- 97.
- 98.
- 99.
- 100.
- 101.
- 102.
- 103.
- 104.
- 105.
- 106.
- 107.
- 108.
- 109.
- 110.
- 111.
- 112.
- 113.
- 114.
- 115.
- 116.
- 117.
- 118.
- 119.
- 120.
- 121.
- 122.
- 123.
- 124.
- 125.
- 126.
- 127.
- 128.
- 129.
- 130.
- 131.
- 132.
- 133.
- 134.
- 135.
- 136.
- 137.
- 138.
- 139.
- 140.
- 141.
- 142.
- 143.
- 144.
- 145.
- 146.
- 147.
- 148.
- 149.
- 150.
- 151.
- 152.
- 153.
- 154.
- 155.
- 156.
- 157.
- 158.
- 159.
- 160.
- 161.
- 162.
- 163.
- 164.
- 165.
- 166.
- 167.
- 168.
- 169.
- 170.
- 171.
- 172.
- 173.
- 174.
- 175.
- 176.
- 177.
- 178.
- 179.
- 180.
- 181.
- 182.
- 183.
- 184.
- 185.
- 186.
- 187.
- 188.
- 189.
- 190.
- 191.
- 192.
- 193.
- 194.
- 195.
- 196.
- 197.
- 198.
- 199.
- 200.
- 201.
- 202.
- 203.
- 204.
- 205.
- 206.
- 207.
- 208.
- 209.
- 210.
- 211.
- 212.
- 213.
- 214.
- 215.
- 216.
- 217.
- 218.
- 219.
- 220.
- 221.
- 222.
- 223.
- 224.
- 225.
- 226.
- 227.
- 228.
- 229.
- 230.
- 231.
- 232.
- 233.
- 234.
- 235.
- 236.
- 237.
- 238.
- 239.
- 240.
- 241.
- 242.
- 243.
- 244.
- 245.
- 246.
- 247.
- 248.
- 249.
- 250.
- 251.
- 252.
- 253.
- 254.
- 255.
- 256.
- 257.
- 258.
- 259.
- 260.
- 261.
- 262.
- 263.
- 264.
- 265.
- 266.
- 267.
- 268.
- 269.
- 270.
- 271.
- 272.
- 273.
- 274.
- 275.
- 276.
- 277.
- 278.
- 279.
- 280.
- 281.
- 282.
- 283.
- 284.
- 285.
- 286.
- 287.
- 288.
- 289.
- 290.
- 291.
- 292.
- 293.
- 294.
- 295.
- 296.
- 297.
- 298.
- 299.
- 300.
- 301.
- 302.
- 303.
- 304.
- 305.
- 306.
- 307.
- 308.
- 309.
- 310.
- 311.
- 312.
- 313.
- 314.
- 315.
- 316.
- 317.
- 318.
- 319.
- 320.
- 321.
- 322.
- 323.
- 324.
- 325.
- 326.
- 327.
- 328.
- 329.
- 330.
- 331.
- 332.
- 333.
- 334.
- 335.
- 336.
- 337.
- 338.
- 339.
- 340.
- 341.
- 342.
- 343.
- 344.
- 345.
- 346.
- 347.
- 348.
- 349.
- 350.
- 351.
- 352.
- 353.
- 354.
- 355.
- 356.
- 357.
- 358.
- 359.
- 360.
- 361.
- 362.
- 363.
- 364.
- 365.
- 366.
- 367.
- 368.
- 369.
- 370.
- 371.
- 372.
- 373.
- 374.
- 375.
- 376.
- 377.
- 378.
- 379.
- 380.
- 381.
- 382.
- 383.
- 384.
- 385.
- 386.
- 387.
- 388.
- 389.
- 390.
- 391.
- 392.
- 393.
- 394.
- 395.
- 396.
- 397.
- 398.
- 399.
- 400.
- 401.
- 402.
- 403.
- 404.
- 405.
- 406.
- 407.
- 408.
- 409.
- 410.
- 411.
- 412.
- 413.
- 414.
- 415.
- 416.
- 417.
- 418.
- 419.
- 420.
- 421.
- 422.
- 423.
- 424.
- 425.
- 426.
- 427.
- 428.
- 429.
- 430.
- 431.
- 432.
- 433.
- 434.
- 435.
- 436.
- 437.
- 438.
- 439.
- 440.
- 441.
- 442.
- 443.
- 444.
- 445.
- 446.
- 447.
- 448.
- 449.
- 450.
- 451.
- 452.
- 453.
- 454.
- 455.
- 456.
- 457.
- 458.
- 459.
- 460.
- 461.
- 462.
- 463.
- 464.
- 465.
- 466.
- 467.
- 468.
- 469.
- 470.
- 471.
- 472.
- 473.
- 474.
- 475.
- 476.
- 477.
- 478.
- 479.
- 480.
- 481.
- 482.
- 483.
- 484.
- 485.
- 486.
- 487.
- 488.
- 489.
- 490.
- 491.
- 492.
- 493.
- 494.
- 495.
- 496.
- 497.
- 498.
- 499.
- 500.
- 501.
- 502.
- 503.
- 504.
- 505.
- 506.
- 507.
- 508.
- 509.
- 510.
6.5 GPS.c代码
接收GPS数据之后进行解析,得到经纬度,方便上传到物联网云平台。
7. 上位机APP开发
为了方便查看地图位置,轨迹等信息,当前采用QT编写了一个配套的上位机,通过华为云IOT的应用侧开发接口,获取设备的影子数据,然后再调用百度地图进行显示目标位置。
接下来就介绍上位机软件的开发流程。
7.1 安装Qt开发环境
Qt是个跨平台的C++开发框架,一份代码支持在不同系统平台编译运行。支持Android、IOS、Windows、Linux等平台。
目前我使用的开发环境是:QT 5.12.6 ,其他版本也可以的。
QT5.12.6的下载地址:
https://download.qt.io/archive/qt/5.12/5.12.6/
打开下载链接后选择下面的版本进行下载:
软件安装时断网安装,否则会提示输入账户。安装的时候,第一个复选框里勾选一个mingw 32编译器即可,其他的不管默认就行,直接点击下一步继续安装。
如果只是在windows下开发,最简单的安装就只选择 MinGW编译器即可,其他的编译器不用勾选。



7.2 应用侧开发接口文档
帮助文档地址: https://support.huaweicloud.com/usermanual-iothub/iot_01_0045.html
当前我这个应用主要是读取设备上传的GPS数据即可,要得到数据有两种方式:
【1】读取设备影子数据,也就是获取设备上传到服务器之后的历史数据(非实时数据)就是设备最后一次传上来的数据,获取影子数据,不需要关心设备是否在线都可以获取。
我这里GPS获取是获取的影子设备数据,也就是得到设备最后一次上传的数据。
地址:https://support.huaweicloud.com/api-iothub/iot_06_v5_0079.html

【2】(查询设备属性)读取实时数据,如果需要设备立即发布当前的数据上来,可以发送同步指令给设备,设备端需要编写解析指令的代码,收到指令后设备根据格式回应数据回来。这种同步实时数据需要设备保持在线才可以响应指令。 具体的使用场景可以根据自己需求设计。
通过查询属性的接口,可以主动请求获取设备详细属性。
流程是:应用层调用这个API接口---->请求服务器----->请求客户端设备------>返回给服务器----->返回给应用层调用处。
文档地址:https://support.huaweicloud.com/api-iothub/iot_06_v5_0034.html

7.3 创建IAM账户
这一步很重要,在开发上位机时,需要调用应用侧的一些接口,这些接口都需要带上token登录密匙。而token登录密匙的生成需要IAM账户才获取。
地址:https://console.huaweicloud.com/iam/?region=cn-north-4#/iam/users
(1)创建用户

(2)填充参数

(3)完成创建

7.4 实现代码
下面贴出请求接口的核心代码。
主要贴出2个比较重要的函数,一个获取token,一个查询设备影子数据。
【本文正在参加物联网有奖征文活动】,活动链接:https://ost.51cto.com/posts/14758
既然社区越来越害怕扶老人,用科技解决也是不错的方法,感谢楼主分享。
mark。