Esp8266做socket实验(TcpServer TcpClient UdpServer Udpclient)

实验环境:

1.  虚拟机:VirtualBox6.0

2.  Ubuntu系统: ubuntu-20.04.1-desktop-amd64

3.  ESP8266: NodeMCU开发板

4.  SDK:V3.4

5.  使用乐鑫SDK提供examples实验使用

 

@0、补充:编译完成后下载出现ttyUSB0找不到的问题:

(图0.00找不到ttyUSB0)

这样需要设置好虚拟机和Ubuntu,然后插拔USB重新挂载:

(图0.10Ubuntu设置USB)

(图0.11拔插USB显示连接上USB)

最后使用命令使Ubuntu挂载USB0:

1.sudo usermod -a -G dialout $USER

2.sudo chmod -R 777 /dev/ttyUSB0

(图0.12可以正常下载和窗口监视)

 

=================================== 分割线 ====================================

@1、Socket预备知识:

  https://blog.csdn.net/pashanhu6402/article/details/96428887

(图1.00Socket在网络中的结构)

@2、TcpServer和 TcpClient

(图2.00TCP协议结构流程图)

TcpServer和 TcpClient流程基本相同,不过TcpServer相对来说多了些流程。复制到工作目录下边。

(图2.10复制Tcp_Server文件到实验目录)

打开Tcp_Server.c文件修改app_main为TcpServer_init,使用CONFIG_EXAMPLE_IPV4而非IPV6。

(图2.11修改app_main方便调用)

(图2.11修改app_main方便调用1)

(图2.12设置端口且使用IPV4)

在app_main函数中调用TcpServer_init完成TcpServer功能

(图2.13调用TcpServer功能)

下载即可使用

@2、tcpclient功能设置与tcpserver基本相同!

@3、udpclient功能设置和udpserver操作也基本相同!!!

	for(;;)
	{
		//使用IPV4
    struct sockaddr_in destAddr;	//处理网络通信地址的结构体(服务器)
    destAddr.sin_addr.s_addr = htonl(INADDR_ANY);	//把本机字节顺序转化为网络字节顺序
    destAddr.sin_family = AF_INET;		//domain:(TCP/IP – IPv4)
    destAddr.sin_port = htons(PORT);		//将主机字节顺序转换为网络字节顺序
    addr_family = AF_INET;		//domain:(TCP/IP – IPv4)
    ip_protocol = IPPROTO_IP;	//#define IPPROTO_IP 0 /* dummy for IP */
    
    inet_ntoa_r(destAddr.sin_addr, addr_str, sizeof(addr_str) - 1);	//转换IP地址到ASIIC字符串的函数
//===================================================================		
    int listen_sock = socket(addr_family, SOCK_STREAM, ip_protocol);	//返回状态码
   
   //ESP_LOGI == printf
    if (listen_sock < 0) {		//返回-1表示失败,其他值参考:https://blog.csdn.net/sstnba/article/details/83380656
        ESP_LOGE(TAG, "Unable to create socket: errno %d", errno);
        break;
    }
    ESP_LOGI(TAG, "Socket created");		//返回正常值说明Socket已经建立起来了
    
    int err = bind(listen_sock, (struct sockaddr *)&destAddr, sizeof(destAddr));	//把一个本地协议地址赋予一个套接字
    //返回值表示绑定操作是否成功,0表示成功, -1表示不成功。函数的返回值千万不要忽视,上次就被人说了。
    if (err != 0) {
        ESP_LOGE(TAG, "Socket unable to bind: errno %d", errno);
        break;
    }
    ESP_LOGI(TAG, "Socket binded");

    err = listen(listen_sock, 1);		//设置监听和队列数
    if (err != 0) {
        ESP_LOGE(TAG, "Error occured during listen: errno %d", errno);
        break;
    }
    ESP_LOGI(TAG, "Socket listening");	//成功监听
//===================================================================		
    //使用IPV4
    struct sockaddr_in sourceAddr;	//处理网络通信地址的结构体(客户端)
    
    uint addrLen = sizeof(sourceAddr);	//地址大小
    int sock = accept(listen_sock, (struct sockaddr *)&sourceAddr, &addrLen);
    if (sock < 0) {
        ESP_LOGE(TAG, "Unable to accept connection: errno %d", errno);
        break;
    }
    ESP_LOGI(TAG, "Socket accepted");  	//阻塞处理-进行握手连接  
//===================================================================		
    
    while (1) //处于监听状态
    {
    	int len = recv(sock, rx_buffer, sizeof(rx_buffer) - 1, 0);	//从TCP连接的另一端接收数据
    	// Error occured during receiving
    	if (len < 0){	//recv在copy时出错了
        ESP_LOGE(TAG, "recv failed: errno %d", errno);
        break;
    	}
    	// Connection closed
    	else if (len == 0) {	//等待协议接收数据时网络中断了
        ESP_LOGI(TAG, "Connection closed");
        break;
   		}
    // Data received
    	else {	//返回值>0,接收实际copy字节数
       	inet_ntoa_r(((struct sockaddr_in *)&sourceAddr)->sin_addr.s_addr, addr_str, sizeof(addr_str) - 1);
       	
       	rx_buffer[len] = 0; // Null-terminate whatever we received and treat like a string
       	ESP_LOGI(TAG, "Received %d bytes from %s:", len, addr_str);
       	ESP_LOGI(TAG, "%s", rx_buffer);

       	int err = send(sock, rx_buffer, len, 0);	//发送回所接收到的数据
       	if (err < 0) {
           ESP_LOGE(TAG, "Error occured during sending: errno %d", errno);
           break;
       	}
     	}
    }
    
    if (sock != -1) {
        ESP_LOGE(TAG, "Shutting down socket and restarting...");
        shutdown(sock, 0);
        close(sock);
    }
	}

技术总结:

@一、注意USB0的挂载问题!

@二、要复制components文件到实验目录,否则会找不到protocol_examples_common.h头文件!

(图3.13网络连接头文件)

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
2022 / 01/ 30: 新版esptool 刷micropython固件指令不是 esptool.py cmd... 而是 esptool cmd... 即可;另外rshell 在 >= python 3.10 的时候出错解决方法可以查看:  已于2022年发布的: 第二章:修复rshell在python3.10出错 免费内容: https://edu.csdn.net/course/detail/29666 micropython语法和python3一样,编写起来非常方便。如果你快速入门单片机玩物联网而且像轻松实现各种功能,那绝力推荐使用micropython。方便易懂易学。 同时如果你懂C语音,也可以用C写好函数并编译进micropython固件里然后进入micropython调用(非必须)。 能通过WIFI联网(2.1章),也能通过sim卡使用2G/3G/4G/5G联网(4.5章)。 为实现语音控制,本教程会教大家使用tensorflow利用神经网络训练自己的语音模型并应用。为实现通过网页控制,本教程会教大家linux(debian10 nginx->uwsgi->python3->postgresql)网站前后台入门。为记录单片机传输过来的数据, 本教程会教大家入门数据库。  本教程会通过通俗易懂的比喻来讲解各种原理与思路,并手把手编写程序来实现各项功能。 本教程micropython版本是 2019年6月发布的1.11; 更多内容请看视频列表。  学习这门课程之前你需要至少掌握: 1: python3基础(变量, 循环, 函数, 常用库, 常用方法)。 本视频使用到的零件与淘宝上大致价格:     1: 超声波传感器(3)     2: MAX9814麦克风放大模块(8)     3: DHT22(15)     4: LED(0.1)     5: 8路5V低电平触发继电器(12)     6: HX1838红外接收模块(2)     7:红外发射管(0.1),HX1838红外接收板(1)     other: 电表, 排线, 面包板(2)*2,ESP32(28)  

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值