目录
前言
user_init
IP定时的回调函数
初始化SNTP
SNTP定时回调函数
前言
介绍了8266如何获取网络时间
user_init
首先将8266设置为sta模式
void ICACHE_FLASH_ATTR user_init(void)
{uart_init(115200,115200); // 初始化串口波特率os_delay_us(10000); // 等待串口稳定os_printf("\r\n=================================================\r\n");os_printf("\t Project:\t%s\r\n", ProjectName);os_printf("\t SDK version:\t%s", system_get_sdk_version());os_printf("\r\n=================================================\r\n");// OLED显示初始化//----------------------------------------------------------------OLED_Init(); // OLED初始化OLED_ShowString(0,0," "); // Internet TimeOLED_ShowString(0,2,"Clock = "); // Clock:时钟OLED_ShowString(0,4,"Temp = "); // Temperature:温度OLED_ShowString(0,6,"Humid = "); // Humidity:湿度//----------------------------------------------------------------LED_Init_JX(); // LED初始化ESP8266_STA_Init_JX(); // ESP8266_STA初始化OS_Timer_IP_Init_JX(1000,1); // 1秒重复定时(获取IP地址)
}
IP定时的回调函数
如果成功获取到ip地址,那么调用ESP8266_SNTP_Init_JX(); 来初始化SNTP
// IP定时的回调函数
//=========================================================================================================
void ICACHE_FLASH_ATTR OS_Timer_IP_cb(void)
{u8 C_LED_Flash = 0; // LED闪烁计次struct ip_info ST_ESP8266_IP; // ESP8266的IP信息u8 ESP8266_IP[4]; // ESP8266的IP地址// 成功接入WIFI【STA模式下,如果开启DHCP(默认),则ESO8266的IP地址由WIFI路由器自动分配】//-------------------------------------------------------------------------------------if( wifi_station_get_connect_status() == STATION_GOT_IP ) // 判断是否获取IP{wifi_get_ip_info(STATION_IF,&ST_ESP8266_IP); // 获取STA的IP信息ESP8266_IP[0] = ST_ESP8266_IP.ip.addr; // IP地址高八位 == addr低八位ESP8266_IP[1] = ST_ESP8266_IP.ip.addr>>8; // IP地址次高八位 == addr次低八位ESP8266_IP[2] = ST_ESP8266_IP.ip.addr>>16; // IP地址次低八位 == addr次高八位ESP8266_IP[3] = ST_ESP8266_IP.ip.addr>>24; // IP地址低八位 == addr高八位// 显示ESP8266的IP地址//------------------------------------------------------------------------------------------------//os_printf("\nESP8266_IP = %d.%d.%d.%d\n",ESP8266_IP[0],ESP8266_IP[1],ESP8266_IP[2],ESP8266_IP[3]);//OLED_ShowIP(24,2,ESP8266_IP); // OLED显示ESP8266的IP地址//------------------------------------------------------------------------------------------------// 接入WIFI成功后,LED快闪3次//----------------------------------------------------for(; C_LED_Flash<=5; C_LED_Flash++){GPIO_OUTPUT_SET(GPIO_ID_PIN(4),(C_LED_Flash%2));delay_ms(100);}os_timer_disarm(&OS_Timer_IP); // 关闭定时器ESP8266_SNTP_Init_JX(); // 初始化SNTP}
}
初始化SNTP
8266最多支持设置三个SNTP服务器,其中SNTP0是主服务器,剩下的两个是备用服务器
我们可以调用这两个API来设置SNTP服务器,参数一是SNTP服务器的优先级
sntp_setservername(0, "us.pool.ntp.org"); // 服务器_0【域名】 参数二是SNTP的域名
sntp_setserver(2, addr); // 服务器_2【IP地址】参数二是SNTP服务器的32位二进制IP地址所以说在使用sntp_setserver(2, addr);之前要先用ipaddr_aton("210.72.145.44", addr); 来将点分十进制字符串形式ip地址转化成32位的二进制ip地址,作为sntp_setserver(2, addr);的参数
之后调用sntp_init(); SNTP初始化API
再设置OS_Timer_SNTP_Init_JX(1000,1); 1秒重复定时(SNTP)
// 初始化SNTP
//=============================================================================
void ICACHE_FLASH_ATTR ESP8266_SNTP_Init_JX(void)
{ip_addr_t * addr = (ip_addr_t *)os_zalloc(sizeof(ip_addr_t));sntp_setservername(0, "us.pool.ntp.org"); // 服务器_0【域名】sntp_setservername(1, "ntp.sjtu.edu.cn"); // 服务器_1【域名】ipaddr_aton("210.72.145.44", addr); // 点分十进制 => 32位二进制sntp_setserver(2, addr); // 服务器_2【IP地址】os_free(addr); // 释放addrsntp_init(); // SNTP初始化APIOS_Timer_SNTP_Init_JX(1000,1); // 1秒重复定时(SNTP)
}
SNTP定时回调函数
调用sntp_get_current_timestamp();来查询当前距离基准时间(1970.01.01 00:00:00 GMT+8)的时间戳(单位:秒) 基准时间是1970年1月1号0时0分0秒东8区
如果TimeStamp这个时间戳不等于0,就说明SNTP获取网络时间成功。
调用sntp_get_real_time(TimeStamp);它的参数就是距离基准时间的时间戳,来获取当前的真实时间。
之后串口打印时间戳,串口打印实际时间
根据获取到的时间来将我们所需要的时间,日期,时刻的值用oled显示,并且oled显示读取到的温湿度数据
// SNTP定时回调函数
//===================================================================================================
void ICACHE_FLASH_ATTR OS_Timer_SNTP_cb(void * arg)
{// 字符串整理 相关变量//------------------------------------------------------u8 C_Str = 0; // 字符串字节计数char A_Str_Data[20] = {0}; // 【"日期"】字符串数组char *T_A_Str_Data = A_Str_Data; // 缓存数组指针char A_Str_Clock[10] = {0}; // 【"时间"】字符串数组char * Str_Head_Week; // 【"星期"】字符串首地址char * Str_Head_Month; // 【"月份"】字符串首地址char * Str_Head_Day; // 【"日数"】字符串首地址char * Str_Head_Clock; // 【"时钟"】字符串首地址char * Str_Head_Year; // 【"年份"】字符串首地址//------------------------------------------------------uint32 TimeStamp; // 时间戳char * Str_RealTime; // 实际时间的字符串// 查询当前距离基准时间(1970.01.01 00:00:00 GMT+8)的时间戳(单位:秒)//-----------------------------------------------------------------TimeStamp = sntp_get_current_timestamp();if(TimeStamp) // 判断是否获取到偏移时间{//os_timer_disarm(&OS_Timer_SNTP); // 关闭SNTP定时器// 查询实际时间(GMT+8):东八区(北京时间)//--------------------------------------------Str_RealTime = sntp_get_real_time(TimeStamp);// 【实际时间】字符串 == "周 月 日 时:分:秒 年"//------------------------------------------------------------------------os_printf("\r\n----------------------------------------------------\r\n");os_printf("SNTP_TimeStamp = %d\r\n",TimeStamp); // 时间戳os_printf("\r\nSNTP_InternetTime = %s",Str_RealTime); // 实际时间os_printf("--------------------------------------------------------\r\n");// 时间字符串整理,OLED显示【"日期"】、【"时间"】字符串//…………………………………………………………………………………………………………………// 【"年份" + ' '】填入日期数组//---------------------------------------------------------------------------------Str_Head_Year = Str_RealTime; // 设置起始地址while( *Str_Head_Year ) // 找到【"实际时间"】字符串的结束字符'\0'Str_Head_Year ++ ;// 【注:API返回的实际时间字符串,最后还有一个换行符,所以这里 -5】//-----------------------------------------------------------------Str_Head_Year -= 5 ; // 获取【"年份"】字符串的首地址T_A_Str_Data[4] = ' ' ;os_memcpy(T_A_Str_Data, Str_Head_Year, 4); // 【"年份" + ' '】填入日期数组T_A_Str_Data += 5; // 指向【"年份" + ' '】字符串的后面的地址//---------------------------------------------------------------------------------// 获取【日期】字符串的首地址//---------------------------------------------------------------------------------Str_Head_Week = Str_RealTime; // "星期" 字符串的首地址Str_Head_Month = os_strstr(Str_Head_Week, " ") + 1; // "月份" 字符串的首地址Str_Head_Day = os_strstr(Str_Head_Month, " ") + 1; // "日数" 字符串的首地址Str_Head_Clock = os_strstr(Str_Head_Day, " ") + 1; // "时钟" 字符串的首地址// 【"月份" + ' '】填入日期数组//---------------------------------------------------------------------------------C_Str = Str_Head_Day - Str_Head_Month; // 【"月份" + ' '】的字节数os_memcpy(T_A_Str_Data, Str_Head_Month, C_Str); // 【"月份" + ' '】填入日期数组T_A_Str_Data += C_Str; // 指向【"月份" + ' '】字符串的后面的地址// 【"日数" + ' '】填入日期数组//---------------------------------------------------------------------------------C_Str = Str_Head_Clock - Str_Head_Day; // 【"日数" + ' '】的字节数os_memcpy(T_A_Str_Data, Str_Head_Day, C_Str); // 【"日数" + ' '】填入日期数组T_A_Str_Data += C_Str; // 指向【"日数" + ' '】字符串的后面的地址// 【"星期" + ' '】填入日期数组//---------------------------------------------------------------------------------C_Str = Str_Head_Month - Str_Head_Week - 1; // 【"星期"】的字节数os_memcpy(T_A_Str_Data, Str_Head_Week, C_Str); // 【"星期"】填入日期数组T_A_Str_Data += C_Str; // 指向【"星期"】字符串的后面的地址// OLED显示【"日期"】、【"时钟"】字符串//---------------------------------------------------------------------------------*T_A_Str_Data = '\0'; // 【"日期"】字符串后面添加'\0'OLED_ShowString(0,0,A_Str_Data); // OLED显示日期os_memcpy(A_Str_Clock, Str_Head_Clock, 8); // 【"时钟"】字符串填入时钟数组A_Str_Clock[8] = '\0';OLED_ShowString(64,2,A_Str_Clock); // OLED显示时间//…………………………………………………………………………………………………………………}// 每5秒,读取/显示温湿度数据//-----------------------------------------------------------------------------------------C_Read_DHT11 ++ ; // 读取DHT11计时if(C_Read_DHT11>=5) // 5秒计时{C_Read_DHT11 = 0; // 计时=0if(DHT11_Read_Data_Complete() == 0) // 读取DHT11温湿度{DHT11_NUM_Char(); // DHT11数据值转成字符串OLED_ShowString(64,4,DHT11_Data_Char[1]); // DHT11_Data_Char[0] == 【温度字符串】OLED_ShowString(64,6,DHT11_Data_Char[0]); // DHT11_Data_Char[1] == 【湿度字符串】}else{OLED_ShowString(64,4,"----"); // Temperature:温度OLED_ShowString(64,6,"----"); // Humidity:湿度}}//-----------------------------------------------------------------------------------------
}
可以看到8266已经成功的获取到了距离基准时间的时间戳,并且根据这个时间戳成功的获取了当前的实际时间 Sun Jul 24 19:14:22 2022 跟电脑系统时间是一致的,这就说明8266通过SNTP成功的获取到了网络时间,在olde上秒一直在走,说明它一直在不停地获取准确的网络时间。