STM32-笔记34-4G遥控灯

news2025/1/8 8:54:10


 

 4G接线

一、项目需求

        服务器通过4G模块远程遥控开关灯。

二、项目实现

复制项目文件夹38-wifi控制风扇项目

重命名为39-4G遥控点灯

打开项目文件

加载文件

main.c

#include "sys.h"
#include "delay.h"
#include "led.h"
#include "uart1.h"
#include "e840.h"
#include "string.h"

int main(void)
{
    HAL_Init();                         /* 初始化HAL库 */
    stm32_clock_init(RCC_PLL_MUL9); /* 设置时钟, 72Mhz */
    led_init();//初始化led灯
    uart1_init(115200);//这里是跟电脑相连接的串口波特率
    e840_init(9600);//我这里的e840的波特率为9600
   // printf("hello word!\r\n");
    
    char recv_data[E840_RX_BUF_SIZE];
    while(1)
    { 
        e840_receive_data(recv_data);//现在是把接收到的数据传到recv_data这里了
        if(strstr(recv_data,"ON") != NULL)//判断我们接收到的字符串里面是否有ON这个值,有的话打卡风扇
            led1_ON();
        else if(strstr(recv_data,"OFF") != NULL)//没有不打开
            led1_OFF();
        delay_ms(10);
//        esp8266_test();
//        delay_ms(500);
    }
}

e840.c

#include "sys.h"
#include "e840.h"
#include "string.h"
#include "stdio.h"
#include "delay.h"
#include "stdarg.h"

uint8_t e840_rx_buf[E840_RX_BUF_SIZE];//定义一个数组,用来保存接收的缓冲区
uint8_t e840_tx_buf[E840_TX_BUF_SIZE];//定义一个数组,用来保存发送的缓冲区
uint16_t e840_cnt = 0,e840_cntPre = 0; //定义一个计数器,和保存计数器原本状态的变量


UART_HandleTypeDef e840_handle = {0};

void e840_uart_init(uint32_t baudrate)
{
    e840_handle.Instance = USART2;
    e840_handle.Init.BaudRate = baudrate;  //波特率
    e840_handle.Init.Mode = UART_MODE_TX_RX;//收发模式;
    e840_handle.Init.Parity = UART_PARITY_NONE;//无校验位
    e840_handle.Init.WordLength = UART_WORDLENGTH_8B;  //字长:8个字长
    e840_handle.Init.StopBits = UART_STOPBITS_1; //停止位:1个停止位
    e840_handle.Init.HwFlowCtl = UART_HWCONTROL_NONE; //无硬件流控
    
    HAL_UART_Init(&e840_handle);
}

void USART2_IRQHandler(void)
{
    uint8_t receive_data = 0;
    //这个函数是用来检查特定的UART接口(在这个例子中是e840_handle所代表的UART接口)是否有数据可读
    if(__HAL_UART_GET_FLAG(&e840_handle,UART_FLAG_RXNE) != RESET)//关注RXNE这个标志位的值是不是不为reset(0)
    {
        if(e840_cnt >= sizeof(e840_rx_buf))//如果接收的字符长度大于字符缓冲区的长度,则把缓冲区长度置0
            e840_cnt = 0;
        //如果RXNE的值为1,证明有数据,所以需要接收数据
        HAL_UART_Receive(&e840_handle,&receive_data,1,1000);//句柄,接收的数据存放在哪?接收数据的个数,超时时间
        e840_rx_buf[e840_cnt++] = receive_data;//将接收的数据存放在e840rx_buf数组中
        
        //HAL_UART_Transmit(&e840_handle,&receive_data,1,1000);//发送数据:句柄,要发送的数据,发送数据的长度,超时
    }
}
//这个函数主要用来判断e840cnt有没有动,如果没有动证明接收完成了
uint8_t e840_wait_receive(void)
{
    if(e840_cnt == 0)//如果cnt为0证明,出现了错误
        return E840_ERROR;//出现错误
    if(e840_cnt == e840_cntPre)//判断当前cnt和上一个cnt是否一致,如果是一致的证明数据不动了,传输完成
    {
        e840_cnt = 0;//cnt清0
        return E840_EOK;//数据接收完成
    }
    e840_cntPre = e840_cnt;//把当前计数器cnt的值赋给之前计数器
    return E840_ERROR;//
     
}
//把接收寄存器的内容清空
void e840_rx_clear(void)
{
    //把接收缓冲器清空
    memset(e840_rx_buf,0,sizeof(e840_rx_buf));
    //清空长度
    e840_cnt = 0;
}
//这个函数在while循环里,来一直判断当前数据是否接收完
uint16_t e840_receive_data(char *recv_data)
{
    if(e840_wait_receive() == E840_EOK)//判断数据是否接受完整
    {
        printf("e840 recv: %s\r\n",e840_rx_buf);//接收完整,打印数据
        //我们把e840_rx_buf中的内容通过memcpy的方式,全部copy到recv_data中
        memcpy(recv_data,e840_rx_buf,strlen((const char *)e840_rx_buf));
        e840_rx_clear();//清除当前接收
        //返回e840_rx_buf的内容以及长度
        return strlen((const char*)recv_data);
    }
    return 0;
}
//发送数据的函数
void e840_send_data(char *fmt,...)
{
    va_list ap;
    uint16_t len;
    va_start(ap,fmt);
    vsprintf((char *)e840_tx_buf,fmt,ap);
    va_end(ap);
    
    len = strlen((const char *)e840_tx_buf);
    HAL_UART_Transmit(&e840_handle,e840_tx_buf,len,100);
}

void e840_init(uint32_t baudrate)
{
    //printf("e840初始化开始...\r\n");
    //e840串口初始化
    e840_uart_init(baudrate);
   
}

e840.h

#ifndef __E840_H__
#define __E840_H__

#include "sys.h"

#define E840_RX_BUF_SIZE         128 //接收的长度
#define E840_TX_BUF_SIZE         64  //发送的长度

#define E840_EOK                  0  //宏定义错误代码 ok
#define E840_ERROR                1  //错误
#define E840_ETIMEOUT             2  //超时
#define E840_EINVAL               3  //数据非法

void e840_init(uint32_t baudrate);
uint16_t e840_receive_data(char *recv_data);
//void e840_test(void);

#endif

注意:

执行结果:

        服务器和4G模块以透传模式相连接,开发板烧录代码,上电,4G模块通电,蓝(电源灯)、黄(模组附着网络灯(SIM))、绿(模组与服务器连接成功)三个指示灯常亮。串口1和电脑通过USB转TTL相连接,并且打开对应串口的串口助手。

        当在服务器中发送信息,可通过透传模式将信息传递给4G模组,4G模组接收到信息,通过RX-TX将信息传递给开发板,该信息执行开发板中代码信息,开发板中的串口1接收到信息,所以,该信息可以在串口助手中查看到,并且,对应LED灯状态改变。

本实验,要注意,串口的波特率和4G模组的波特率,可在main函数中使用正确的波特率。

补充:在本实验中,使用花生壳APP可以将我们的一个内网的IP地址,映射成我们的外网IP地址,然后我们的其他外网的ip就可以与该外网进行合法通信,这个过程叫做内网穿透。

  外网连接(内网穿透)

 内网连接

三、出现的问题

        当使用亿佰特时,选择对应的串口号和波特率之后,打开串口,进入配置,显示进入AT指令模式失败原因?

波特率选择的不对,换个波特率试试

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

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

相关文章

游戏引擎学习第77天

仓库: https://gitee.com/mrxiao_com/2d_game 回顾昨天的 bug 今天我们继续开发进度,进行调试昨天代码的问题,主要是关于如何跟踪玩家和敌人在世界中的高度位置。虽然我们做的是一款 2D 游戏,但我们希望能够处理多层的房间,玩家…

STM32完全学习——使用定时器1精确延时

一、定时器的相关配置 首先一定要是递减定时器,递增的不太行,控制的不够准确,其次在大于10微秒的延时是非常准确的,小于的话,就没有那没准,但是凑合能用。误差都在一个微秒以内。使用高级定时器也就是时钟…

aardio —— 虚表 —— 模拟属性框

写了个简单的属性框例程,抛砖引玉,期待你做出更丰富强大的功能。 本例演示:折叠子行、选择框、输入文本、输入数值、下拉选择、选择图片、选择颜色、选择字体等功能。 只有想不到,没有做不到,发挥你的想象力吧。 imp…

[微服务]redis主从集群搭建与优化

搭建主从集群 单节点Redis的并发能力是有上限的,要进一步提高Redis的并发能力,就需要搭建主从集群,实现读写分离。 1. 主从集群结构 下图就是一个简单的Redis主从集群结构: 如图所示,集群中有一个master节点、两个s…

设计模式 行为型 观察者模式(Observer Pattern)与 常见技术框架应用 解析

观察者模式(Observer Pattern)是一种行为设计模式,它定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态发生变化时,会通知所有观察者对象,使它们能够自动更新。 一…

《Opencv》图像的旋转

一、使用numpy库实现 np.rot90(img,-1) 后面的参数为-1时事顺时针旋转,为1时是逆时针旋转。 import cv2 import numpy as np img cv2.imread(./images/kele.png) """方法一""" # 顺时针90度 rot_1 np.rot90(img,-1) # 逆时针90度…

Android Studio 安装配置(个人笔记)

Android studio安装的前提是必须保证安装了jdk1.8版本以上 一、查看是否安装jdk cmd打开命令行,输入java -version 最后是一个关键点 输入 javac ,看看有没有相关信息 没有就下载jdk Android studio安装的前提是必须保证安装了jdk1.8版本以上 可以到…

spicy.signal 报错解决

报错: ImportError: cannot import name ‘kaiser’ from ‘scipy.signal’ 解决办法 找到import的位置:将 from scipy.signal import kaiser 修改为 from scipy.signal.windows import kaiser

学习threejs,导入AWD格式的模型

👨‍⚕️ 主页: gis分享者 👨‍⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍⚕️ 收录于专栏:threejs gis工程师 文章目录 一、🍀前言1.1 ☘️THREE.AWDLoader AWD模型加…

【Dify】Dify自定义模型设置 | 对接DMXAPI使用打折 Openai GPT 或 Claude3.5系列模型方法详解

一、Dify & DMXAPI 1、Dify DIFY(Do It For You)是一种自动化工具或服务,旨在帮助用户简化操作,减少繁琐的手动操作,提升工作效率。通过DIFY,用户能够快速完成任务、获取所需数据,并且可以…

5. CSS引入方式

5.1 CSS的三种样式 按照 CSS 样式书写的位置(或者引入的方式),CSS样式表可以分为三大类: 1.行内样式表(行内式) 2.内部样式表(嵌入式) 3. 外部样式表(链接式) 5.2 内部样式表 …

大型语言模型(LLM)中的tokens是什么

大型语言模型(LLM)中的tokens是什么 在大型语言模型(LLM)中,tokens是文本处理的基本单位,它可以是一个单词、一个字符、一个标点符号,或者是一个特殊的标记。以下是关于tokens的详细介绍及举例: 一、tokens的定义和作用 定义:tokens是将文本分割成的一个个有意义的…

计算机网络 (29)网络地址转换NAT

前言 网络地址转换(Network Address Translation,NAT)是计算机网络中的一种重要协议,它主要用于将私有IP地址转换为公共IP地址,以实现内部网络与外部网络之间的通信。 一、基本概念 NAT是一种在局域网(LAN&…

Node.js——fs(文件系统)模块

个人简介 👀个人主页: 前端杂货铺 🙋‍♂️学习方向: 主攻前端方向,正逐渐往全干发展 📃个人状态: 研发工程师,现效力于中国工业软件事业 🚀人生格言: 积跬步…

【cuda学习日记】2.1 2D matrix操作

2.1.1 检查块和线程索引 #include <stdio.h> #include <stdlib.h> #include <time.h> #include <string.h> #include <cuda_runtime.h>#define CHECK(call) \{\const cudaError_t error call; \if (error ! cudaSuccess)\{\printf("Error…

Nginx:会话保持

会话保持 是指在负载均衡环境中,确保来自同一用户的多个请求都发送到同一个后端服务器。这通常用于那些需要记住用户状态或上下文的应用程序,例如购物车、登录状态等。 会话保持的重要性 用户体验:保证用户在整个会话期间的一致性体验,避免因不同服务器间的数据不同步导致…

Java-数据结构-链表-高频面试题(1)

在上一篇文章中&#xff0c;我们学习了链表中的"单向链表"&#xff0c;但学可不代表就是学会了&#xff0c;能够运用链表的地方比比皆是&#xff0c;解题方法也是层出不穷&#xff0c;今天就让我们巩固一下"单向链表"的知识吧~ 第一题&#xff1a;相交链表…

5. 多线程(3) --- synchronized

文章目录 前言1. 如何解决线程安全问题 [回顾]2. synchronized 关键字2.1. 示例2.2.对示例进行变化2.3 synchronized的其他写法2.4 synchronized的特性2.4.1 互斥2.4.2. 刷新内存2.4.3. 可重入 前言 前面我们通过在两个线程中共同对count进行加一操作&#xff0c;最后得到的结…

阿尔法linux开发板ping不通百度

我使用的阿尔法linux板子&#xff0c;发现按照《03【正点原子】I.MX6U网络环境TFTP&NFS搭建手册V1.3.2》一套操作下来&#xff0c;还是没办法实现板子上网。 我总结了下面方法&#xff0c;我如何实现联网和互ping通&#xff0c;大致总结下三步 一、pc端的wifi网络&#xf…

Qt之屏幕录制设计(十六)

Qt开发 系列文章 - screencap&#xff08;十六&#xff09; 目录 前言 一、实现原理 二、实现方式 1.创建录屏窗口 2.录屏窗口类定义 3.自建容器对象定义 4.用户使用 5.效果演示 总结 前言 利用Qt实现屏幕录制设计&#xff0c;可以通过使用Qt自带的类QScreen、QPixma…