ESP8266获取网络时间 实时时钟

news2024/11/25 4:24:41

程序现象

 

一、用串口调试助手调试

1.发送指令AT+RST重启模块使应用模式更改生效;

2.发送指令ATE0取消回显

3.使用串口发送指令AT+CWMODE=1设置模块Wi-Fi应用模式为Station模式;

4.发送指令AT+CWJAP ="ssid","pwd"连接AP;

5.发送指令AT+CIPMUX=0设置模块为单路连接模式,模块默认为单路连接模式;

6.发送指令AT+CIPSTART="TCP","api.k780.com",80 与服务器建立TCP连接 ;

7.发送指令AT+CIPMODE=1设置模块传输模式为透传模式;

8.发送指令AT+CIPSEND开启透传模式向服务器发送数据,模块收到此指令后先换行后返回“>”

9.剩下只需要发送 GET http://api.k780.com:88/?app=life.time&appkey=10003&sign=b59bc3ef6191eb9f747dd4e83c99f2a4&format=json&HTTP/1.1\r\n,就能获取到年月时间。      (调用一个网络API接口,这个接口会返回标准网络时间)

 可以看到发送完第九条指令后会收到如上图,所以整体思路:用串口进行通信,发送AT指令,获取到的数据存在Buf中,然后对Buf中的数据解析。获取timestamp后为时间戳,获取datetime_1后为日期时间。

USART串口模块

这一部分的初始化和之前一样,串口接收部分要进行判断如果接收到 '>' ,用Flag标记表示后面调用API后会收到时间戳和时间等数据。

#include "stm32f10x.h"                  // Device header
#include "MyUSART.h"
#include "esp.h"
#include <string.h>

char Buf[512];
unsigned char i;
uint8_t Flag;


void MyUSART_Init(void)
{
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 ,ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA ,ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStructure;                
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOA,&GPIO_InitStructure);
	
	USART_InitTypeDef USART_InitStructure;              
	USART_InitStructure.USART_BaudRate = 115200;
	USART_InitStructure.USART_HardwareFlowControl =USART_HardwareFlowControl_None;
	USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
	USART_InitStructure.USART_Parity =USART_Parity_No ;
	USART_InitStructure.USART_StopBits = USART_StopBits_1;
	USART_InitStructure.USART_WordLength = USART_WordLength_8b;
	USART_Init (USART1,&USART_InitStructure);
	
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
	NVIC_InitTypeDef NVIC_InitStructure;
	NVIC_InitStructure.NVIC_IRQChannel =USART1_IRQn;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
	NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
	NVIC_Init(&NVIC_InitStructure);
	
	USART_ITConfig(USART1,USART_IT_RXNE,ENABLE);
	USART_Cmd(USART1,ENABLE);
	
}
void MyUSART_SendString(char* str)
{
	uint8_t stri=0;
	while(str[stri] != '\0')
		USART_SendData (USART1,str[stri++]);
}
void USART1_IRQHandler()
{
	if(USART_GetITStatus(USART1,USART_IT_RXNE))
	{
		Buf[i++]=USART_ReceiveData(USART1);
		if(Buf[i-1] == '>')
		{
			Flag=1;
		}
		else if((Buf[i-2]=='\r')&&(Buf[i-1]=='\n'))  
		{
			Buf[i-2]='\0';
			i = 0;
		}
		USART_ClearITPendingBit(USART1,USART_IT_RXNE);
	}
}

ESP8266WIFI模块

这一部分初始化操作发送AT指令,在主函数中根据返回的数字可以判断哪里有问题

Comman_Rec()这个函数用来处理Buf中接收到的数据。

#include "stm32f10x.h"                  // Device header
#include "MyUSART.h"
#include <stdio.h>
#include <string.h>
#include "Delay.h"
#include "OLED.H"
#include <stdlib.h>

extern char Buf[512];
const char* WIFI ="GT";
const char* PASS="123456789";
extern uint8_t Flag;
uint32_t timestamp_cnt;
char subdatetime_1[11]; 
char subdatetime_2[8];
uint32_t temp = 2;

int fputc(int ch,FILE *f )   //printf重定向  
{
	USART_SendData(USART1,(uint8_t)ch);
	while(USART_GetFlagStatus (USART1,USART_FLAG_TC) == RESET);
	return ch;
}
char esp_Init(void)
{

	memset(Buf,0,sizeof(Buf));
	printf("AT+RST\r\n");
	Delay_ms(2000);

	memset(Buf,0,sizeof(Buf));
	printf("ATE0\r\n");    //关闭回显memset(Buf,0,sizeof(Buf));
	Delay_ms(1000);
	if(strcmp(Buf,"OK")!=0 )
		return 1;
	
	memset(Buf,0,sizeof(Buf));
	printf("AT+CWMODE=1\r\n");
	Delay_ms(50);
	if(strcmp(Buf,"OK")!=0)
		return 2;
	
	
	memset(Buf,0,sizeof(Buf));
	printf("AT+CWJAP=\"%s\",\"%s\"\r\n",WIFI,PASS); //连接热点
	Delay_ms(1000);
	if(strcmp(Buf,"OK")!=0)
		return 3;
	
	memset(Buf,0,sizeof(Buf));
	printf("AT+CIPMUX=0\r\n");
	Delay_ms(500);
	if(strcmp(Buf,"OK")!=0)
		return 4;
	
	memset(Buf,0,sizeof(Buf));
	printf("AT+CIPSTART=\"TCP\",\"api.k780.com\",80\r\n");
	Delay_ms(500);
	if(strcmp(Buf,"OK")!=0)
		return 5;
	
	memset(Buf,0,sizeof(Buf));
	printf("AT+CIPMODE=1\r\n");
	Delay_ms(500);
	if(strcmp(Buf,"OK")!=0)
		return 6;
		
	memset(Buf,0,sizeof(Buf));
	printf("AT+CIPSEND\r\n");
	Delay_ms(500);
	if(Flag != 1)
		return 7;
	return 0;
}

void ESP_Pub(void)        //当接收到'>'时
{
	if(Flag == 1)
	{
		printf("GET http://api.k780.com:88/?app=life.time&appkey=10003&sign=b59bc3ef6191eb9f747dd4e83c99f2a4&format=json&HTTP/1.1\r\n");
		Delay_ms(2000);
	}
}


void Comman_Rec(void)
{
	    char *timestamp_start = strstr(Buf,"timestamp"); // 定位到"timestamp"字段的起始位置
		if (timestamp_start != NULL) 
		{
			timestamp_start += (strlen("timestamp")+3); // 跳过"timestamp"
			char timestamp[11];
			strncpy(timestamp, timestamp_start, 10); // 复制10个字符到timestamp数组
			timestamp[10] = '\0'; 					// 添加字符串结束符
			timestamp_cnt =atoi(timestamp);		//转换为整数
		}
		char *datetime1_start = strstr(Buf,"datetime_1"); 
		if (datetime1_start != NULL) 
		{
			datetime1_start += (strlen("datetime_1")+3); 
			strncpy(subdatetime_1, datetime1_start, 10); 
			subdatetime_1[10] = '\0'; 
			strncpy(subdatetime_2, datetime1_start+11, 7); 
			subdatetime_2[7] = '\0'; 

		}
}

主函数

#include "stm32f10x.h"                  // Device header
#include "OLED.H"
#include <stdio.h>
#include "Delay.h"
#include "MyUSART.H" 
#include "esp.h"

extern char Buf[512];
extern uint8_t Flag;
extern uint32_t timestamp_cnt;
extern char subdatetime_1[11];
extern char subdatetime_2[9];


void Init(void)
{
	uint8_t Judge=0;
	OLED_Init();
	OLED_ShowString(1,1,"Linking..");
	Delay_ms(100);
	MyUSART_Init(); 
	
	do
	{
		Judge = esp_Init();
		OLED_ShowString(1,1,"code:    ");
		OLED_ShowNum(2,1,Judge,1);
	}while(Judge!=0);  

}

int main(void)
{
	Init();
	if(Flag==1)
	{
		OLED_ShowString(1,1,"SUCCESS");
	}
	else
	{
		OLED_ShowString(1,1,"FAIL");
	}
	while(1)
	{
		ESP_Pub();
		Comman_Rec();
		OLED_ShowNum(1,1,timestamp_cnt,10);
		OLED_ShowString(2,1,subdatetime_1);
		OLED_ShowString(3,1,subdatetime_2);

	}

}

注意:这个代码存在的问题是比标准的北京时间慢两秒,因为每次调用网络API接口都会Delay2秒,所以可以只获取时间戳,加上两秒后再根据网上的代码将时间戳转换为具体的时间格式。还可以根据上一节实时时钟的代码实现掉电不丢失的功能。而且有时候显示一会会跳变一下,也有很多问题,我在网上没找到stm32库函数的这个功能实现,还没找到更好的思路。

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

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

相关文章

react-native-webview RN和html双向通信

rn登录后得到的token需要传递给网页&#xff0c;js获取到的浏览器信息需要传递给rn RN Index.js: import React from react import { WebView } from react-native-webview import useList from ./useListexport default function Index(props) {const { uri, jsCode, webVie…

23款奔驰S450 4MATIC升级车载冰箱系统,快乐就是这么朴实无华呀

凉爽餐饮随时触手可及。容积10升的可拆卸冷藏箱与后排扶手和谐融合。如此一来&#xff0c;即使在炎炎夏日&#xff0c;也可享受沁凉的冷饮。

Vue3头像(Avatar)

效果如下图&#xff1a;在线预览 APIs 参数说明类型默认值必传shape指定头像的形状‘circle’ | ‘square’‘circle’falsesize设置头像的大小number | ‘large’ | ‘small’ | ‘default’ | Responsive‘default’falsesrc图片类头像资源地址string‘’falsealt图片无法显…

关于微信临时文件wxfile://tmp文件如何处理,微信小程序最新获取头像和昵称

分享-2023年资深前端进阶&#xff1a;前端登顶之巅-最全面的前端知识点梳理总结&#xff0c;前端之巅 *分享一个使用比较久的&#x1fa9c; 技术栈&#xff1a;taro框架 vue3版本 解决在微信小程序获取微信头像时控制台报错&#xff1a;找不着wxfile://tmp 文件路径,失败&…

迁移协调器 - 就地迁移模式

在本系列博客的第一部分中&#xff0c;我们从高层级视角介绍了 Migration Coordinator 提供的所有模式&#xff0c;Migration Coordinator 是内置于 NSX 中的完全受 GSS 支持的工具&#xff0c;可将 NSX for vSphere 迁移到 NSX (NSX-T)。 本系列的第二篇博客将详细介绍就地迁…

cesium 卫星环绕扫描

成果图 源码 let viewer new Cesium.Viewer(cesiumContainer,{// terrainProvider: Cesium.createWorldTerrain(),geocoder: false, // 隐藏查找位置homeButton: false, // 隐藏返回视角到初始位置sceneModePicker: false, // 隐藏视角模式的选择baseLayerPicker: false, // 隐…

亚马逊云科技助力珠海丹德构建安全技术底座,促进商业发展

随着消费者对商品质量和安全关注度的不断提高&#xff0c;防伪、溯源、防窜已经成为企业关注的重要领域。据前瞻产业研究院数据显示&#xff0c;2028年中国防伪行业市场容量将超过4000亿元&#xff0c;未来市场对防伪、溯源、防窜技术的需求和重视程度可见一斑。 作为一家用智慧…

软件测试项目实战,电商业务功能测试点汇总(全覆盖)

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 支付功能怎么测试…

医院后勤管理用什么系统好?的修医院报修管理系统有哪些优势?

随着医院后勤工作量的不断增加&#xff0c;需要协调和维护的设备和部门也随之增多。传统的医院后勤管理方式已经显得不够优越&#xff0c;其劣势日益凸显&#xff0c;无法满足实际工作需求。因此&#xff0c;快速推动医院后勤信息化管理已成为当前医院发展的迫切需求。而的修医…

进销存记账软件2023排行榜,秦丝、智慧记、管家婆哪家更好用?

进销存记账软件已经成为很多实体店必备的一款软件&#xff0c;使用进销存记账软件可以帮助实体店解决手工记账效率低下、对账麻烦且出错率高等问题。 很多实体店都是小本生意&#xff0c;选择进销存记账软件时由于缺乏经验&#xff0c;随意选择&#xff0c;结果买回来之后一堆问…

无人机跟随一维高度避障场景--逻辑分析

无人机跟随一维高度避障场景--逻辑分析 1. 源由2. 视频3. 问题3.1 思维发散3.2 问题收敛 4. 图示4.1 水平模式4.2 下坡模式4.3 上坡模式4.4 碰撞分析 5. 总结6. 参考资料 1. 源由 最近拿到一台测试样机&#xff0c;功能很多&#xff0c;就不多赘述。 这里针对跟随功能进行下吐…

java面试基础 -- 方法重载 方法重写

目录 重载 重写 重载 方法的重载是指在同一个类中定义多个方法, 他们具有相同的名称, 但是具有不同的参数列表, 例如: public void myMethod(int arg1) {// 方法体 }public void myMethod(int arg1, int arg2) {// 方法体 }public void myMethod(String arg1) {// 方法体 }…

企业权限管理(九)-用户操作

用户操作 1用户查询 UserController findAll Controller RequestMapping("/user") public class UserController {Autowiredprivate IUserService userService;RequestMapping("/findAll.do")public ModelAndView findAll() throws Exception {ModelAndVie…

PMP考试通过标准是什么?

PMP 新考纲一共是 180道题&#xff0c;答对 108道就通过了&#xff0c;具体怎么看通过没有&#xff1f; 一、查看是否通过 1、登录PMI 官网&#xff0c;点击“Log In” 如果忘记 PMI 的账号、密码了也别着急&#xff0c;去找你报名的培训机构&#xff0c;一般报名处有记录&…

C语言 — qsort 函数

介绍&#xff1a;qsort是一个库函数&#xff0c;用来对数据进行排序&#xff0c;可以排序任意类型的数据。 void qsort &#xff08;void*base&#xff0c; size_t num, size_t size, int(*compart)(const void*,constvoid*) &#xff09; qsort 具有四个参数&#xff1a; …

分割等和子集——力扣416

思路:动态规划 bool canPartition(vector<int>& nums){int n=nums.size(

建筑师设计师太难了,既要学BIM、无人机,还要学GIS!

我&#xff0c;一个平平无奇的城市规划专业&#xff08;建筑专业、路桥专业&#xff09;大学生&#xff0c;还有一年要毕业&#xff0c;很担心工作以后受到社会的毒打&#xff0c;遂问导师和学长&#xff0c;我要自学点什么技能和软件&#xff1f; 学长A&#xff1a;CAD&#x…

mobile wireless network

老人家教学&#xff1a;手机设置无线网络 以我家里无线网络为例&#xff1a;FJGDWL-zeng 以苹果手机为例&#xff1a;其他手机类似操作 1&#xff09;【设置】 2&#xff09;【无线局域网】 3&#xff09;【无线局域网】列表&#xff0c;有的话&#xff0c;直接选中&#xff0…

攸信技术:厦门火炬高新区产业联合会的新伙伴!

8月12日&#xff0c;2023年度厦门火炬高新区产业联合会会员大会在厦门国际会议中心酒店举行。来自火炬高新区产业联合会各领域的200余家优秀企业代表齐聚一堂&#xff0c;共商合作&#xff0c;共谋发展。 本次大会攸信技术作为新会员单位参加厦门火炬高新区产业联合会会员授牌仪…