完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
一、ESP32 WIFI介绍
Wi-Fi 库支持配置及监控 ESP32 Wi-Fi 连网功能。 支持配置:
二、WiFi 的启动(STA 及 AP 模式) 1. WiFi STA 模式(连接到其他设备的热点) (1) 步骤及API简介 ① 初始化nvs_flash ② 初始化esp_netif ③ 创建事件循环event_loop —— event_loop是esp32库的一种事件处理模式,中文曰“事件循环” ④ 初始化、配置WiFi并启动 ⑤ 事件处理(使用第③步创建的事件循环) (2) 代码示例 ① 初始化nvs_flash #include ... ... /*尝试初始化一次nvs_flash*/ esp_err_t err = nvs_flash_init(); if(err == ESP_ERR_NVS_NO_FREE_PAGES || err == ESP_ERR_NVS_NEW_VERSION_FOUND){ /*第一次初始化失败, 检查错误, 处理错误, 重新初始化*/ nvs_flash_erase(); err = nvs_flash_init(); } ESP_ERROR_CHECK(err); ... ... ② 初始化esp_netif #include "esp_netif.h" ____________ esp_netif_t *pEsp_wifi_netif; esp_netif_init(); /*创建wifi sta模式的默认netif, 返回一个指针*/ pEsp_wifi_netif = esp_netif_create_default_wifi_sta(); /* (非必须) 这行代码是修改ESP32的主机名, 即WiFi设备名, 连接其他WiFi时显示的名称 */ esp_netif_set_hostname(esp_wifi_netif, "Augtons");// 不建议使用汉字 上述代码第 6 行函数esp_netif_create_default_wifi_sta()是必要的,只不过函数返回值可以在不需要的时候忽略(本文我们借助这个创建得到的esp_netif对象来修改了主机名为Augtons,因此保留了这个返回值) ③ 初始化event_loop #include "esp_event.h" _____________________ esp_event_loop_create_default(); esp_event_handler_instance_register(WIFI_EVENT, ESP_EVENT_ANY_ID, 想要用于处理事件的函数, NULL, &wifi_handler); esp_event_handler_instance_register(IP_EVENT, IP_EVENT_STA_GOT_IP, 想要用于处理时间的函数, NULL, &ip_handler)); 代码解析:
[tr]函数名esp_event_handler_instance_register()[/tr]
④ 初始化、配置WiFi并启动
#include "driver/wifi.h" /* 第一步, WiFi初始化 */ /* 这个宏 WIFI_INIT_CONFIG_DEFAULT() 可以初始化一 个wifi_init_config(wifi初始化配置)结构体为默认值 */ wifi_init_config_t wifi_init_config = WIFI_INIT_CONFIG_DEFAULT(); esp_wifi_init(&wifi_init_config); /* 第二步, WiFi配置 */ wifi_config_t wifi_config = { .sta = { .ssid = "WiFi_SSID", // wifi名(ssid) .password = "password",// wifi密码 } }; esp_wifi_set_config(WIFI_IF_STA, &wifi_config); esp_wifi_set_mode(WIFI_MODE_STA); // 设置工作模式 /* 第三步, 启动WiFi */ esp_wifi_start(); ⑤ 事件处理(使用第③步创建的事件循环) 这里这个函数就是第 ③ 步注册的事件循环handler函数。 handler函数模板如下: void xxx(void *arg, esp_event_base_t eventBase, int32_t eventID, void *eventData) { /* 函数参数一:arg。表示传递给handler函数的参数,在第三步register函数里声明 函数参数二,eventBase,表示事件基,详见第三步函数esp_event_handler_instance_register()解析 函数参数三:eventID,表示事件ID,详见第三步函数esp_event_handler_instance_register()解析 函数参数四,表示传递给这个事件的数据。例如: 例如事件基 IP_EVENT 下的 IP_EVENT_STA_GOT_IP 事件会把获取到的IP地址传递过来,见下方示例。 */ } 下面为WiFi事件处理的一个简单的handler函数。 处理的事件: [tr]事件基事件含义[/tr]
void wifi_event_handler(void *arg, esp_event_base_t eventBase, int32_t eventID, void *eventData){ ip_event_got_ip_t *ip = eventData; if(eventBase == WIFI_EVENT){ switch (eventID) { default: break; case WIFI_EVENT_STA_START: printf("WIFI_STARTED!n"); esp_wifi_connect(); break; case WIFI_EVENT_STA_DISCONNECTED: ESP_LOGI("user", "Connect failedn"); esp_wifi_connect(); break; } }else if(eventBase == IP_EVENT){ if(eventID == IP_EVENT_STA_GOT_IP){ printf("connected, got ip: "IPSTR"n", IP2STR(&ip->ip_info.ip)); //dosomething //例如:使用任务通知xTaskNotifyGive(任务句柄);来唤醒一个任务等 } } } 当WIFI_EVENT_STA_START发生即WiFi启动成功之后,我们就连接WiFi。 当WIFI_EVENT_STA_DISCONNECT发生即WiFi断开连接/连接失败后,我们就尝试重新连接 当IP_EVENT_STA_GOT_IP发生即WiFi连接成功并获取到IP地址之后,我们就在屏幕上打印出获取到的IP地址 printf("connected, got ip: "IPSTR"n", IP2STR(&ip->ip_info.ip)); 然后做其他的事情 2. WiFi AP 模式(ESP32作为热点供其他设备链接) AP模式的WiFi与STA模式极其类似: 不同的是,在进行WiFi配置时wifi_config_t结构体用法会改变。 wifi_config_t wifi_config = { .ap = { .ssid = "your_ssid", .ssid_len = strlen("your_ssid"), .channel = EXAMPLE_ESP_WIFI_CHANNEL, .password = "password", .max_connection = EXAMPLE_MAX_STA_CONN, .authmode = WIFI_AUTH_WPA_WPA2_PSK }, }; 剩余的代码,读者可以参考 ESP-IDF 的官方示例 三、STA模式下扫描外部WiFi 1. 步骤 在上一步WiFi启动之后,我们就能进行扫描附近的WiFi了 这里需要了解 3 个相关的API函数 esp_wifi_scan_start() esp_wifi_scan_get_ap_num() esp_wifi_scan_get_ap_records() [tr]函数名esp_wifi_scan_start()[/tr]
[tr]函数名esp_wifi_scan_get_ap_num()[/tr]
[tr]函数名esp_wifi_scan_get_ap_records()[/tr]
2. 代码示例 TaskHandle_t wifi_scan_handle; // 上边的这个变量储存了创建的wifi扫描任务句柄 // 对应下边示例的函数 task_wifi_scan(void *arg) void wifi_event_handler(void *arg, esp_event_base_t eventBase, int32_t eventID, void *eventData){ ... if(eventBase == WIFI_EVENT){ if(eventID == WIFI_EVENT_STA_START){ // printf("WiFi 已启动"); // 向WiFi扫描任务发送任务通知,通知WiFi启动成功,可以开始扫描了 (这里的任务通知相当于一个二值信号量) xTaskNotifyGive(wifi_scan_handle); } } } void task_wifi_scan(void *arg){ // 等待来自event_handler发来的任务通知,在等到通知之前,此任务进入阻塞状态。知道接收到WiFi启动成功的通知之后才开始往下执行。 ulTaskNotifyTake(pdTRUE, portMAX_DELAY); uint16_t num; wifi_ap_record_t records[8]; esp_wifi_scan_start(NULL, true); esp_wifi_scan_get_ap_num(&num); if(num > 8){ num = 8; // if 的目的是如果扫描到了8个以上的WiFi,则只保留前8个 } esp_wifi_scan_get_ap_records(&num, records); if(num > 8){ num = 8; // if 的目的同上 } // 这个for循环是用来将RSSI信号强度,转换成信号等级。 // 如 rssi_level == 5 时,表示WiFi满格 for(int i = 0; i < num; i++){ uint8_t rssi_level = 0; switch (records.rssi) { case -100 ... -88: rssi_level = 1; break; case -87 ... -77: rssi_level = 2; break; case -76 ... -66: rssi_level = 3; break; case -65 ... -55: rssi_level = 4; break; default: if(records.rssi < -100){ rssi_level = 0; }else{ rssi_level = 5; } break; } // 逐条打印扫描到的WiFi printf("—————【第 %2d 个WiFi】———————n", i+1); printf("WiFi名称: %sn", records.ssid); printf("信号强度: %d格n", rssi_level); printf("WiFi: 安全类型: %dnn", records.authmode); } vTaskDelete(NULL); } 运行结果图: |
||
|
||
只有小组成员才能发言,加入小组>>
3309 浏览 9 评论
2988 浏览 16 评论
3490 浏览 1 评论
9050 浏览 16 评论
4085 浏览 18 评论
1171浏览 3评论
602浏览 2评论
const uint16_t Tab[10]={0}; const uint16_t *p; p = Tab;//报错是怎么回事?
594浏览 2评论
用NUC131单片机UART3作为打印口,但printf没有输出东西是什么原因?
2331浏览 2评论
NUC980DK61YC启动随机性出现Err-DDR是为什么?
1894浏览 2评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-12-20 04:20 , Processed in 1.117322 second(s), Total 79, Slave 60 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号