一、连接WIFI

在进行时间同步之前,先连接WIFI

#include "wifi.h"

#include <string.h>
#include <stdlib.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "freertos/event_groups.h"
#include "esp_wpa2.h"
#include "esp_event.h"
#include "esp_log.h"
#include "esp_system.h"
#include "nvs_flash.h"
#include "esp_netif.h" static wifi_status_t wifi_status = WIFI_STATUS_DEINIT; // WIFI 默认为未初始化状态
static EventGroupHandle_t s_wifi_event_group; // WIFI 事件组
esp_event_handler_instance_t instance_any_id; // 注册事件处理程的标识
esp_event_handler_instance_t instance_got_ip; // /* 事件组允许有多个事件标志位 */
static const int WIFI_STATUS_INIT_BIT = BIT0; // WIFI 初始化状态
static const int WIFI_STATUS_START_BIT = BIT1; // WIFI 启动状态
static const int WIFI_STATUS_SCAN_BIT = BIT3; // WIFI 扫描状态
static const int WIFI_STATUS_CONNECTED_BIT = BIT4; // WIFI 连接状态 /* 需要连接的WIFI名称和密码 */
static wifi_sta_config_t cfg_sta = {
.ssid = "test",
.password = "test8888",
}; /* 互斥锁 */
SemaphoreHandle_t xWifiSemaphore; static const char *TAG = "WIFI"; /**
* @brief WIFI 事件
*
* @param arg
* @param event_base
* @param event_id
* @param event_data
*/
static void wifi_event_handler(void* arg, esp_event_base_t event_base,
int32_t event_id, void* event_data)
{
/* WIFI事件 */
if (event_base == WIFI_EVENT)
{
switch (event_id)
{
/* WIFI 就绪 */
case WIFI_EVENT_WIFI_READY:
ESP_LOGI(TAG,"ESP32 WiFi ready");
break;
/* WIFI 扫描完成 */
case WIFI_EVENT_SCAN_DONE:
ESP_LOGI(TAG,"ESP32 finish scanning AP");
xEventGroupSetBits(s_wifi_event_group, WIFI_STATUS_SCAN_BIT); // 设置WIFI标志位为打开状态
break;
/* WIFI 启动 */
case WIFI_EVENT_STA_START:
ESP_LOGI(TAG,"ESP32 station start");
xEventGroupSetBits(s_wifi_event_group, WIFI_STATUS_START_BIT); // 设置WIFI标志位为打开状态
wifi_status = WIFI_STATUS_START; // 设置WIFI为打开状态
break;
/* WIFI 关闭 */
case WIFI_EVENT_STA_STOP:
ESP_LOGI(TAG,"ESP32 station stop");
xEventGroupClearBits(s_wifi_event_group, WIFI_STATUS_START_BIT); // 设置WIFI标志位为关闭状态
wifi_status = WIFI_STATUS_STOP; // 设置WIFI为关闭状态
break;
/* WIFI 连接成功 */
case WIFI_EVENT_STA_CONNECTED:
ESP_LOGI(TAG,"ESP32 station connected to AP");
xEventGroupSetBits(s_wifi_event_group, WIFI_STATUS_CONNECTED_BIT); // 设置WIFI标志位为关闭状态
wifi_status = WIFI_STATUS_CONNECTED; // 设置WIFI为关闭状态
break;
/* WIFI 断开连接 */
case WIFI_EVENT_STA_DISCONNECTED:
ESP_LOGI(TAG,"ESP32 station disconnected from AP");
xEventGroupClearBits(s_wifi_event_group, WIFI_STATUS_CONNECTED_BIT); // 设置WIFI标志位为关闭状态
wifi_status = WIFI_STATUS_DISCONNECTED; // 设置WIFI为关闭状态
break;
/* WIFI 接入点认证方式改变 */
case WIFI_EVENT_STA_AUTHMODE_CHANGE:
ESP_LOGI(TAG,"the auth mode of AP connected by ESP32 station changed");
break; default:
ESP_LOGI(TAG,"Other status");
break;
}
} if(event_base == IP_EVENT && event_id == IP_EVENT_STA_GOT_IP) {
ip_event_got_ip_t* event = (ip_event_got_ip_t*) event_data;
ESP_LOGI(TAG, "Got IP: " IPSTR, IP2STR(&event->ip_info.ip));
}
} /**
* @brief WIFI 初始化
*
*/
void app_wifi_init(void)
{
// 初始化 NVS
esp_err_t ret = nvs_flash_init();
if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND)
{
ESP_ERROR_CHECK(nvs_flash_erase());
ret = nvs_flash_init();
}
ESP_ERROR_CHECK(ret); /* 初始化底层TCP/IP堆栈 */
ESP_ERROR_CHECK(esp_netif_init()); s_wifi_event_group = xEventGroupCreate(); // 创建新的事件组
ESP_ERROR_CHECK(esp_event_loop_create_default()); // 创建默认事件循环
esp_netif_t *sta_netif = esp_netif_create_default_wifi_sta(); // 创建默认WIFI STA
assert(sta_netif); /* 使用默认参数初始化WIFI */
wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
ESP_ERROR_CHECK( esp_wifi_init(&cfg) ); /* 向默认循环注册事件处理程序 */
ESP_ERROR_CHECK( esp_event_handler_instance_register(WIFI_EVENT, ESP_EVENT_ANY_ID, &wifi_event_handler, NULL, instance_any_id) );
ESP_ERROR_CHECK( esp_event_handler_instance_register(IP_EVENT, IP_EVENT_STA_GOT_IP, &wifi_event_handler, NULL, instance_got_ip) ); ESP_ERROR_CHECK( esp_wifi_set_mode(WIFI_MODE_STA) ); // 设置WIFI模式 xEventGroupSetBits(s_wifi_event_group, WIFI_STATUS_INIT_BIT); // 设置WIFI标志位为初始化状态
wifi_status = WIFI_STATUS_INIT; // 设置WIFI状态为初始化状态
xWifiSemaphore = xSemaphoreCreateMutex(); // 创建互斥锁
} void app_main(void)
{
app_wifi_init();
ESP_ERROR_CHECK( esp_wifi_start() ); // 启动WIFi /* 而直接将wifi_sta_config_t(或指针)转为wifi_config_t(或指针)是GCC的拓展语法,如下 */
esp_wifi_set_config(WIFI_IF_STA, (wifi_config_t *) &cfg_sta); esp_wifi_connect();
}

二、时间同步

/**
* @brief 同步时间完成后的回调函数
*
* @param tv
*/
void time_update_callback(struct timeval* tv)
{
/* 设置时区 */
setenv("TZ", "CST-8", 1);
tzset();
} /**
* @brief 在联网的情况下获取网络时间
*
*/
static void get_network_time(void)
{
/* 等待 WIFI 连接*/
EventBits_t bits = xEventGroupWaitBits(s_wifi_event_group, WIFI_STATUS_CONNECTED_BIT, pdFALSE, pdTRUE, portMAX_DELAY); /* SNTP校时 */
sntp_setoperatingmode(SNTP_OPMODE_POLL);
sntp_setservername(0, "ntp.aliyun.com");
sntp_set_time_sync_notification_cb(time_update_callback);
sntp_init(); // 启动校时
}

注意:不使用sntp_set_time_sync_notification_cb()注册同步成功回调时,也可以使用sntp_get_sync_status()轮询检测同步是否完毕

三、获取系统时间

static struct tm timeinfo = {0};           // 时间寄存器
time_t now = 0; time(&now);
localtime_r(&now, &timeinfo); /* 打印获取到的时间 */
char str[64];
strftime(str, sizeof(str), "%c", &timeinfo);
ESP_LOGI(TAG, "time updated: %s", str); ESP_LOGI(TAG, "%d%d:%d%d", timeinfo.tm_hour / 10, timeinfo.tm_hour % 10, timeinfo.tm_min / 10, timeinfo.tm_min % 10);
ESP_LOGI(TAG, "%d-%d-%d", timeinfo.tm_year + 1900, timeinfo.tm_mon + 1, timeinfo.tm_mday); switch (timeinfo.tm_wday)
{
case 0:
ESP_LOGI(TAG, "星期日");
break;
case 1:
ESP_LOGI(TAG, "星期一");
break;
case 2:
ESP_LOGI(TAG, "星期二");
break;
case 3:
ESP_LOGI(TAG, "星期三");
break;
case 4:
ESP_LOGI(TAG, "星期四");
break;
case 5:
ESP_LOGI(TAG, "星期五");
break;
case 6:
ESP_LOGI(TAG, "星期六");
break;
default:
break;
}

参考文献

ESP32 之 ESP-IDF 教学(二十)—— SNTP校时:<https://blog.csdn.net/m0_50064262/article/details/126690030

ESP32 SNTP校时的更多相关文章

  1. GPS时钟装置,北斗卫星授时,ntp校时服务器,网络时间服务器

    GPS时钟装置,北斗卫星授时,ntp校时服务器,网络时间服务器 一.GPS时钟装置产品特点:技术交流-岳峰 15901092122:Q522508213 GPS时钟装置是针对计算机.自动化装置等进行校 ...

  2. NTP校时设置

    一.Windows Server 2008 – Time Server 前言: 国家时间与频率标准实验室  && NTP服务器 也可以忽略1~6 直接跳7 如果已改过机码请使用 1   ...

  3. 记一个有想法却没能力实现的硬件产品——mp3校时闹钟

    枕头旁的闹钟,我想大家都用过,很便宜.用一节干电池供电.但其最大的缺点就是不太准,不能校时. 电池啥事用光,也不知道.钟是走的很慢,没按时闹,搞的自己迟了到. 于是就有了我的漫长思考过程... 先说手 ...

  4. Linux NTP校时

    1.安装客户端(root权限运行) apt-get install ntpdate 2.修改配置文件:“/etc/default/ntpdate”,     NTPSERVERS="ntp. ...

  5. 北斗时钟同步系统-GPS卫星授时设备-NTP网络校时服务器

    北斗时钟同步系统-GPS卫星授时设备-NTP网络校时服务器 北斗时钟同步系统-GPS卫星授时设备-NTP网络校时服务器 北斗时钟同步系统-GPS卫星授时设备-NTP网络校时服务器 论述当下网络时间同步 ...

  6. GPS校时器,GPS时钟装置,NTP网络时间服务器

    GPS校时器,GPS时钟装置,NTP网络时间服务器 GPS校时器,GPS时钟装置,NTP网络时间服务器 GPS校时器,GPS时钟装置,NTP网络时间服务器 GPS校时器,GPS时钟装置,NTP网络时间 ...

  7. 网络时钟服务器,NTP授时设备,北斗网络校时服务器,GPS时间同步器

    网络时钟服务器,NTP授时设备,北斗网络校时服务器,GPS时间同步器 网络时钟服务器,NTP授时设备,北斗网络校时服务器,GPS时间同步器 论述当下网络时间同步的重要性   北京华人开创科技发展有限公 ...

  8. 实现基于NTP协议的网络校时功能

    无论PC端还是移动端系统都自带时间同步功能,基于的都是NTP协议,这里使用C#来实现基于NTP协议的网络校时功能(也就是实现时间同步). 1.NTP原理 NTP[Network Time Protoc ...

  9. c# 系统校时工具类

    using System; using System.Collections.Generic; using System.Linq; using System.Text; namespace HuaT ...

  10. GPS北斗NTP校时服务器原理及功能介绍

    在科技的发展下GPS北斗NTP校时服务器也得到了广泛应用,比如工业.科研.航空航天.公共场所等领域都用到了GPS北斗NTP校时服务器,该时间服务器以卫星时间为基准授时准确,替代了传统钟表授时的单一和时 ...

随机推荐

  1. 记录--uniapp 应用APP跳转微信小程序

    这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 最近APP项目开发完成,在评审会上老板提了一个需求,想在开发的APP上添加一个链接,可以跳转公司的小程序商城. 原以为会很复杂,结果只有短 ...

  2. KingbaseES Insert On Conflict 功能

    针对数据写入时有主键冲突的情况,INSERT ON CONFLICT语法可以将冲突主键的INSERT行为转换为UPDATE行为,从而实现冲突主键的覆盖写入.该特性又称UPSERT覆盖写,与MySQL的 ...

  3. OGNL表达式注入分析

    OGNL基础 依赖 <dependency> <groupId>ognl</groupId> <artifactId>ognl</artifact ...

  4. C++ 简单实现shared_ptr

    共享指针 管理指针的存储,提供有限的垃圾回收工具,并可能与其他对象共享该管理. shared_ptr类型的对象都能够获得指针的所有权并共享该所有权:一旦它们获得所有权,当最后一个所有者释放该所有权时, ...

  5. Unity中国、Cocos为OpenHarmony游戏生态插上腾飞的翅膀

      2023年是OpenHarmony游戏生态百花齐放的一年!为了扩展OpenHarmony游戏生态,OpenHarmony在基金会成立了游戏SIG小组,游戏SIG小组联合cocos,从cocos2d ...

  6. Hi3861 通过UART串口协议与其它开发板进行通信

    一.搭建编译环境 1.下载虚拟机VMware和Ubuntu20.0.14 下载 VMware Workstation Pro | CN https://www.vmware.com/cn/produc ...

  7. RPM打包教程

    一.rpm是什么 rpm是一种安装包的格式.就像在Windows系统上我们常见的安装包格式是exe和msi一样,在linux上常见的安装包格式是deb和rpm.一般在红帽系列的系统上,不支持deb,所 ...

  8. Linux-搭建内网yum源

    部署要求: 服务器:CentOS7 YUM源:阿里云 空间要求:CentOS6+CentOS7 50G,考虑后期更新预留,LVS空间100G 1.在服务器配置CentOS7的yum源和CentOS6的 ...

  9. c# 优化代码的一些规则——判断null值得不同写法[六]

    前言 先来看一个例子: 假设updated 是一个委托: if(updated!=null) { updated(); } 请问上面写法在多线程中安全吗?如果不安全会抛出什么异常呢? 正文 上面的答案 ...

  10. Pytorch DistributedDataParallel(DDP)教程二:快速入门实践篇

    一.简要回顾DDP 在上一篇文章中,简单介绍了Pytorch分布式训练的一些基础原理和基本概念.简要回顾如下: 1,DDP采用Ring-All-Reduce架构,其核心思想为:所有的GPU设备安排在一 ...