ESP32 开发笔记(三)源码示例 20_WIFI_STA_TCP_Server 在站模式STA下实现TCP服务端

野性酷女 2022-12-07 04:29 341阅读 0赞

开发板购买链接

https://item.taobao.com/item.htm?spm=a2oq0.12575281.0.0.50111deb2Ij1As&ft=t&id=626366733674

开发板简介
开发环境搭建 windows

ESP32例程为C语言开发,并非Python/Arduino/AT指令开发,基于ESP-IDF_V4.2框架库,采用纯C语言开发,开发工具为Visual Studio Code

Android(安卓)例程全部为原生Android开发,例程全部支持到SDK版本29 Android 10.0(Q) 开发工具为Andirod Studio 4.0

PC源码例程为Visual Studio 2013开发,C++语言,基于MFC

送的MQTT账号,仅用于同学们开发测试,大家不要用到实际产品上,不定期更换密码,会在QQ群公布

以下列出的例程源码均编写完成(不断增加中),开发教程正在编写
基础例程:
0_Hello Bug (ESP_LOGX与printf) 工程模板/打印调试输出
1_LED LED亮灭控制
2_LED_Task 使用任务方式控制LED
3_LEDC_PWM 使用LEDC来控制LED实现呼吸灯效果
4_ADC_LightR 使用ADC读取光敏电阻实现光照传感
5_KEY_Short_Long 按钮长按短按实现
6_TouchPad_Interrupt 电容触摸中断实现
7_WS2812_RMT 使用RMT实现RGB_LED彩虹变色示例
8_DHT11_RMT 使用RMT实现读取DHT11温湿度传感器
9_SPI_SDCard 使用SPI总线实现TF卡文件系统示例
10_IIC_ADXL345 使用IIC总线实现读取ADXL345角度加速度传感器
11_IIC_AT24C02 使用IIC总线实现小容量数据储存测试
12_IR_Rev_RMT 使用RMT实现红外遥控接收解码(NEC编码)
13_IR_Send_RMT 使用RMT实现红外数据发送(NEC编码)
14_WIFI_Scan 附近WIFI信号扫描示例
15_WIFI_AP 创建软AP示例
16_WIFI_AP_TCP_Server 在软AP模式下实现TCP服务端
17_WIFI_AP_TCP_Client 在软AP模式下实现TCP客户端
18_WIFI_AP_UDP 在软AP模式下实现UDP通讯
19_WIFI_STA 创建STA站模连接路由器
20_WIFI_STA_TCP_Server 在站模式STA下实现TCP服务端
21_WIFI_STA_TCP_Client 在站模式STA下实现TCP客户端
22_WIFI_STA_UDP 在站模式STA下实现UDP通讯
23_LCD_Test LCD液晶触摸屏显示测试
24_XPT2046_Touch_Test 电阻触摸XPT2046驱动测试触摸校正
ESP32应用例程
Bluetooth_RGBLight Android手机通过蓝牙控制RGB灯变色
Bluetooth_Sensor Android手机通过蓝牙读取开发板姿态和温度湿度参数
Bluetooth_Test Android手机通过蓝牙与开发板通讯
Encryption_MD5 MD5加密
Encryption_SHA1 SHA1加密
Encryption_SHA256 SHA256加密
EncryptionDecrypt_AES_CBC AES_CBC加密
EncryptionDecrypt_AES_ECB AES_ECB加密
ESP32_SmartConfig 通过Android手机配置WIFI连网
HTTP_GET_Request_Weather HTTP Get请求天气预报
JSON_Package 创建JSON数据包
JSON_Parsing 解析JSON数据包
LCD JPEG_Effect 解码JGEG图片特效显示
MQTT MQTT通讯测试
Android(安卓)工具源码 ESP32_Bluetooth_Chat Android通过蓝牙与开发板通讯测试
ESP32_Bluetooth_RGBLight Android通过蓝牙控制开发板RGB灯
ESP32_Bluetooth_Sensor Android通过蓝牙读取开发板3D姿态和温度湿度数据
ESP32_SmartConfig Android手机配置WIFI配网
MQTT_Test Android与开发板通过MQTT协议通讯
TCP_Client Android作为TCP客户端与开发板通讯
UDP_Client Android通过UDP协议与开发板通讯
PC(Visual Studio 2013 C++ MFC)工具源码
MQTT_调试助手 PC通过MQTT与开发板通讯
TCP 调试助手 PC通过TCP服务器/客户端与开发板通讯
UDP 调试助手 PC通过UDP与开发板通讯
LVGL源码
LVGL_Full_Test LVGL官方例程(使用到了大分部控件)
LVGL_Arc_Test 圆弧指示器通过触摸动态改变值
LVGL_Bar_Test 通过动画方式动态演示Bar进度条的使用方法
LVGL_Button_Test 通过5种不同动态效果的按钮学习按钮的创建与使用
LVGL_ButtonMatrix_Test 创建一个计算器布局来演示矩阵按钮控件的使用
LVGL_Calendar_Test 日历控件的创建和使用
LVGL_Canvas_Test 透明画布和画一个圆角过渡色矩形并旋转角度学习画布的使用
LVGL_Chart_Test 通过创建三个不同风格的图表来学习图表控件的使用
LVGL_Checkbox_Test 学习复选框控件的创建和使用
LVGL_ColorPicker_Test 创建一个颜色选择器并动态显示当前颜色的RGB值
LVGL_Container_Test 在内容控件上动态创建三个文本标签学习内容控件的自适应布局
LVGL_Drop_down_List_Test 创建三个不同类型的下拉选择控件
LVGL_Gauge_Test 创建一个动态仪表和静态多指针仪表
LVGL_Image_Test 通过四个滑动条控件控制图片颜色的变化来学习图像控件的使用
LVGL_ImageButton_Test 创建一个图片背景的图像按钮
LVGL_Keyboard_Test 通过一个文本输入框控件还学习键盘控件的调用关闭设置
LVGL_Label_Test 创建内容颜色可变,长文本滚动展示,带阴影3D效果三个标签控件
LVGL_LED_Test 创建三个LED,学习LED控件的调光,颜色,开关的设置
LVGL_Line_Test 通过二维数组创建一段折线来演示线控件的使用
LVGL_LineMeter_Test 创建两个不同的线段弧形指示器动态展示数据
LVGL_List_Test 创建一个带图标的列表控件
LVGL_MessageBox_Test 创建一个带按钮的消息框控件
LVGL_ObjectMask_Test 创建一个变幻色的文本学习蒙版遮罩效果
LVGL_Page_Test 学习页面控件的使用
LVGL_Roller_Test 通过示例学习滑动列表选择器控件
LVGL_Slider_Test 创建一个单向和一个双向滑动条控件
LVGL_Spinbox_Test 学习微调控件的使用
LVGL_Spinner_Test 创建三个不同的环形加载器
LVGL_Switch_Test 创建两个不同的开关控件
LVGL_Table_Test 创建一个简单的表格
LVGL_Tabview_Test 实现三页的页面切换学习Tabview控件
LVGL_Textarea_Test 长按实现打字机效果的Textarea控件
LVGL_Tileview_Test 实现四面环形触摸切换的Tileview控件
LVGL_Window_Test 创建一个窗口,带设置子窗口学习窗口控件的使用

Station模式简介

Station模式又叫做站点工作模式,类似于无线终端

处于Station模式下的ESP32,可以连接到AP(WIFI路由器)。通过Station(简称为“STA”)模式,ESP32作为客户端连接到路由的wifi信号。

基于AP组建的基础无线网络(Infra):Infra:也称为基础网,是由AP创建,众多STA加入所组成的无线网络,这种类型的网络的特点是AP是整个网络的中心,网络中所有的通信都通过AP来转发完成。

在此模式下设备可以通过AP分配的IP地址直接访问外网和内网,原理图如下:

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2NuaWNmaG51aQ_size_16_color_FFFFFF_t_70

TCP介绍

传输控制协议(TCP,Transmission Control Protocol)是一种面向连接的、可靠的、基于字节流的传输层通信协议。

TCP旨在适应支持多网络应用的分层协议层次结构。 连接到不同但互连的计算机通信网络的主计算机中的成对进程之间依靠TCP提供可靠的通信服务。TCP假设它可以从较低级别的协议获得简单的,可能不可靠的数据报服务。 原则上,TCP应该能够在从硬线连接到分组交换或电路交换网络的各种通信系统之上操作。

TCP分为服务端Server和客户端Client,服务端创建服务等待客户端连接,客户端连接后即可与服务端相互发送消息,服务端只有一个,客户端可以是N个,可以同时连接服务端,服务器不能主动去连接客户端,必须客户端主动连接服务端才能相互发送消息。

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2NuaWNmaG51aQ_size_16_color_FFFFFF_t_70 1

实验流程

1、ESP32创建站模式连接WIFI

2、连接成功后ESP32创建TCP服务端

3、电脑端创建TCP Client(电脑必须与开发板在同一路由器下)

4、相互发送数据

一、编写代码

先引用必要头文件

  1. #include <stdio.h>
  2. #include "esp_system.h"
  3. #include "esp_spi_flash.h"
  4. #include "esp_wifi.h"
  5. #include "esp_event.h"
  6. #include "esp_log.h"
  7. #include "esp_err.h"
  8. #include "nvs_flash.h"
  9. #include "esp_event.h"
  10. #include <string.h>
  11. #include <sys/socket.h>
  12. #include "freertos/FreeRTOS.h"
  13. #include "freertos/task.h"
  14. #include "freertos/event_groups.h"
  15. #include "esp_wifi.h"
  16. #include "driver/gpio.h"

编写主函数,先创建STA模式并连接WIFI,连接成功后按下BOOT键开始创建TCP服务端任务,等待客户端连接

  1. // 主函数
  2. void app_main(void)
  3. {
  4. ESP_LOGI(TAG, "APP Start......");
  5. //初始化flash
  6. esp_err_t ret = nvs_flash_init();
  7. if (ret == ESP_ERR_NVS_NO_FREE_PAGES){
  8. ESP_ERROR_CHECK(nvs_flash_erase());
  9. ret = nvs_flash_init();
  10. }
  11. ESP_ERROR_CHECK(ret);
  12. wifi_init_sta();// WIFI作为STA的初始化
  13. gpio_pad_select_gpio(LED_GPIO);// 选择要操作的GPIO
  14. gpio_set_direction(LED_GPIO, GPIO_MODE_OUTPUT);// 设置GPIO为推挽输出模式
  15. while(1){
  16. vTaskDelay(100 / portTICK_RATE_MS);
  17. if(gpio_get_level(0)==0){
  18. //新建一个tcp连接任务
  19. xTaskCreate(&tcp_connect, "tcp_connect", 4096, NULL, 5, NULL);
  20. break;
  21. }
  22. }
  23. while(1) {
  24. gpio_set_level(LED_GPIO, 0);// GPIO输出低
  25. vTaskDelay(500 / portTICK_PERIOD_MS);
  26. gpio_set_level(LED_GPIO, 1);// GPIO输出高
  27. vTaskDelay(500 / portTICK_PERIOD_MS);
  28. }
  29. }

修改WIFI名称和密码,修改为家里的WIFI

  1. #define WIFI_SSID "TP-YIXIN" // WIFI 网络名称
  2. #define WIFI_PAS "a12345678" // WIFI 密码

创建STA模式并连接

  1. // WIFI作为STA的初始化
  2. void wifi_init_sta()
  3. {
  4. tcp_event_group = xEventGroupCreate();
  5. tcpip_adapter_init();
  6. ESP_ERROR_CHECK(esp_event_loop_init(event_handler, NULL));
  7. wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
  8. ESP_ERROR_CHECK(esp_wifi_init(&cfg));
  9. wifi_config_t wifi_config = {
  10. .sta = {
  11. .ssid = WIFI_SSID,
  12. .password = WIFI_PAS},
  13. };
  14. ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA));
  15. ESP_ERROR_CHECK(esp_wifi_set_config(ESP_IF_WIFI_STA, &wifi_config));
  16. ESP_ERROR_CHECK(esp_wifi_start());
  17. ESP_LOGI(TAG, "wifi_init_sta finished.");
  18. ESP_LOGI(TAG, "connect to ap SSID:%s password:%s \n",WIFI_SSID, WIFI_PAS);
  19. }

创建TCP 服务端任务

  1. // 建立TCP连接并从TCP接收数据
  2. static void tcp_connect(void *pvParameters)
  3. {
  4. while (1){
  5. g_rxtx_need_restart = false;
  6. // 等待WIFI连接信号量,死等
  7. xEventGroupWaitBits(tcp_event_group, WIFI_CONNECTED_BIT, false, true, portMAX_DELAY);
  8. ESP_LOGI(TAG, "start tcp connected");
  9. TaskHandle_t tx_rx_task = NULL;
  10. vTaskDelay(3000 / portTICK_RATE_MS);// 延时3S准备建立server
  11. ESP_LOGI(TAG, "create tcp server");
  12. int socket_ret = create_tcp_server(true);// 建立server
  13. if (socket_ret == ESP_FAIL){// 建立失败
  14. ESP_LOGI(TAG, "create tcp socket error,stop...");
  15. continue;
  16. }else{// 建立成功
  17. ESP_LOGI(TAG, "create tcp socket succeed...");
  18. // 建立tcp接收数据任务
  19. if (pdPASS != xTaskCreate(&recv_data, "recv_data", 4096, NULL, 4, &tx_rx_task)){
  20. ESP_LOGI(TAG, "Recv task create fail!");
  21. }else{
  22. ESP_LOGI(TAG, "Recv task create succeed!");
  23. }
  24. }
  25. while (1){
  26. vTaskDelay(3000 / portTICK_RATE_MS);
  27. if (g_rxtx_need_restart){// 重新建立server,流程和上面一样
  28. ESP_LOGI(TAG, "tcp server error,some client leave,restart...");
  29. // 建立server
  30. if (ESP_FAIL != create_tcp_server(false)){
  31. if (pdPASS != xTaskCreate(&recv_data, "recv_data", 4096, NULL, 4, &tx_rx_task)){
  32. ESP_LOGE(TAG, "tcp server Recv task create fail!");
  33. }else{
  34. ESP_LOGI(TAG, "tcp server Recv task create succeed!");
  35. g_rxtx_need_restart = false;//重新建立完成,清除标记
  36. }
  37. }
  38. }
  39. }
  40. }
  41. vTaskDelete(NULL);
  42. }

创建TCP服务端

  1. // 建立tcp server
  2. esp_err_t create_tcp_server(bool isCreatServer)
  3. {
  4. //首次建立server
  5. if (isCreatServer){
  6. ESP_LOGI(TAG, "server socket....,port=%d", TCP_PORT);
  7. server_socket = socket(AF_INET, SOCK_STREAM, 0);//新建socket
  8. if (server_socket < 0){
  9. show_socket_error_reason("create_server", server_socket);
  10. close(server_socket);//新建失败后,关闭新建的socket,等待下次新建
  11. return ESP_FAIL;
  12. }
  13. //配置新建server socket参数
  14. server_addr.sin_family = AF_INET;
  15. server_addr.sin_port = htons(TCP_PORT);
  16. server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
  17. //bind:地址的绑定
  18. if (bind(server_socket, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0){
  19. show_socket_error_reason("bind_server", server_socket);
  20. close(server_socket);//bind失败后,关闭新建的socket,等待下次新建
  21. return ESP_FAIL;
  22. }
  23. }
  24. //listen,下次时,直接监听
  25. if (listen(server_socket, 5) < 0){
  26. show_socket_error_reason("listen_server", server_socket);
  27. close(server_socket);//listen失败后,关闭新建的socket,等待下次新建
  28. return ESP_FAIL;
  29. }
  30. //accept,搜寻全连接队列
  31. connect_socket = accept(server_socket, (struct sockaddr *)&client_addr, &socklen);
  32. if (connect_socket < 0){
  33. show_socket_error_reason("accept_server", connect_socket);
  34. close(server_socket);//accept失败后,关闭新建的socket,等待下次新建
  35. return ESP_FAIL;
  36. }
  37. ESP_LOGI(TAG, "tcp connection established!");
  38. return ESP_OK;
  39. }

创建接收处理数据任务

  1. // 接收数据任务
  2. void recv_data(void *pvParameters)
  3. {
  4. int len = 0;
  5. char databuff[1024];
  6. while (1){
  7. memset(databuff, 0x00, sizeof(databuff));//清空缓存
  8. len = recv(connect_socket, databuff, sizeof(databuff), 0);//读取接收数据
  9. g_rxtx_need_restart = false;
  10. if (len > 0){
  11. ESP_LOGI(TAG, "recvData: %s", databuff);//打印接收到的数组
  12. send(connect_socket, databuff, strlen(databuff), 0);//接收数据回发
  13. //sendto(connect_socket, databuff , sizeof(databuff), 0, (struct sockaddr *) &remote_addr,sizeof(remote_addr));
  14. }else{
  15. show_socket_error_reason("recv_data", connect_socket);//打印错误信息
  16. g_rxtx_need_restart = true;//服务器故障,标记重连
  17. vTaskDelete(NULL);
  18. }
  19. }
  20. close_socket();
  21. g_rxtx_need_restart = true;//标记重连
  22. vTaskDelete(NULL);
  23. }

WIFI事件处理函数

  1. // wifi 事件
  2. static esp_err_t event_handler(void *ctx, system_event_t *event)
  3. {
  4. switch (event->event_id)
  5. {
  6. case SYSTEM_EVENT_STA_START: //STA模式-开始连接
  7. esp_wifi_connect();
  8. break;
  9. case SYSTEM_EVENT_STA_DISCONNECTED: //STA模式-断线
  10. esp_wifi_connect();
  11. xEventGroupClearBits(tcp_event_group, WIFI_CONNECTED_BIT);
  12. break;
  13. case SYSTEM_EVENT_STA_CONNECTED: //STA模式-连接成功
  14. xEventGroupSetBits(tcp_event_group, WIFI_CONNECTED_BIT);
  15. break;
  16. case SYSTEM_EVENT_STA_GOT_IP: //STA模式-获取IP
  17. ESP_LOGI(TAG, "got ip:%s\n",ip4addr_ntoa(&event->event_info.got_ip.ip_info.ip));xEventGroupSetBits(tcp_event_group, WIFI_CONNECTED_BIT);
  18. break;
  19. default:
  20. break;
  21. }
  22. return ESP_OK;
  23. }

二、下载测试

打开ESP-IDF Command Prompt

cd命令进入此工程目录

  1. cd F:\ESP32_DevBoard_File\20_WIFI_STA_TCP_Server

查看电脑设备管理器中开发板的串口号

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2NuaWNmaG51aQ_size_16_color_FFFFFF_t_70 2

执行idf.py -p COM9 flash monitor从串口9下载并运行打开口显示设备调试信息 Ctrl+c退出运行

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L2NuaWNmaG51aQ_size_16_color_FFFFFF_t_70 3

测试流程

当开发板成功连接WIFI后会打印WIFI_STA_TCP_Client Demo: got ip:192.168.XXX.XXX

开发板按BOOT键开始创建TCP Server

打开电脑端网络助手

网络助手选择TCP Client

服务器地址:输入开发板获取到的IP

远程主机端口:9527

电脑端发送数据,开发板会原样返回数据

发表评论

表情:
评论列表 (有 0 条评论,341人围观)

还没有评论,来说两句吧...

相关阅读