实验目的:
         通过 Socket  编程实现  pyWiFi-ESP32-S3  与电脑服务器助手建立连接,相互收  
  发数据。 
  首先先来简单了解一下Socket
我们先来看看网络层级模型图,这是构成网络通信的基础:

         我们看看 TCP/IP  模型的传输层和应用层,传输层比较熟悉的概念是  TCP  和 UDP, UPD  协议基本就没有对  IP  层的数据进行任何的处理了。而  TCP  协议还加入 了更加复杂的传输控制,比如滑动的数据发送窗口(Slice Window ),以及接收确 认和重发机制,以达到数据的可靠传送。应用层中网页常用的则是 HTTP 。那么我们先来解析一下这 TCP  和  HTTP  两者的关系。  
           我们知道网络通信是最基础是依赖于 IP  和端口的, HTTP  一般情况下默认使用端口 80。举个简单的例子:我们逛淘宝,浏览器会向淘宝网的网址(本质是IP)和端口发起请求,而淘宝网收到请求后响应,向我们手机返回相关网页数据信息,实现了网页交互的过程。而这里就会引出一个多人连接的问题,很多人访问淘宝网,实际上接收到网页信息后就断开连接,否则淘宝网的服务器是无法支撑这么多人长时间的连接的,哪怕能支持,也非常占资源。 
           也就是应用层的  HTTP  通过传输层进行数据通信时, TCP  会遇到同时为多个应用程序进程提供并发服务的问题。多个 TCP  连接或多个应用程序进程可能需要通过同一个 TCP  协议端口传输数据。为了区别不同的应用程序进程和连接,许多计算机操作系统为应用程序与 TCP / IP  协议交互提供了套接字 (Socket)接口。应用层可以和传输层通过 Socket 接口,区分来自不同应用程序进程或网络连接的通信,实现数据传输的并发服务。 
           简单来说,Socket  抽象层介于传输层和应用层之间, 跟  TCP/IP  并没有必然的联系。Socket  编程接口在设计的时候,就希望也能适应其他的网络协议。  
  
         套接字(socket)是通信的基石,是支持 TCP/IP 协议的网络通信的基本操作单元。它是网络通信过程中端点的抽象表示,包含进行网络通信必须的五种信息:  
           连接使用的协议(通常是 TCP  或  UDP ),本地主机的  IP  地址,本地进程的协议端  
  口,远地主机的  IP  地址,远地进程的协议端口。 
          所以,socket  的出现只是可以更方便的使用  TCP/IP  协议栈而已,简单理解就是其对 TCP/IP  进行了抽象,形成了几个最基本的函数接口。比如  create , listen, accept, connect , read  和  write  等等。以下是通讯流程: 
  
重点
 从上图可以看到,建了  Socket  通信需要一个服务器端和一个客户端,以本实验为例,pyWiFi-ESP32-S3  作为客户端,电脑使用网络调试助手作为服务器端,双方使用 TCP  协议传输。 
   对于客户端,则需要知道电脑端的  IP  和端口即可建立连接。(端口可以自定义,范0~65535 ,注意不占用常用的  80  等端口即可。)  
   以上的内容,简单来说就是如果用户面向应用来说,那么  ESP32-S3  只需要知道通讯协议是  TCP  或  UDP 、服务器的  IP  和端口号 这  3  个信息,即可向服务器发起连接和发送信息。就这么简单。 
  代码如下
'''
实验名称:Socket通讯
版本:v1.0
日期:2024.4
作者:冰美式
说明:通过Socket编程实现pyWiFi-ESP32-S3与电脑服务器助手建立TCP连接,相互收发数据
可以控制舵机。
'''#导入相关模块
import network,usocket,time
from machine import SoftI2C,Pin,Timer, PWM
from ssd1306 import SSD1306_I2C#初始化相关模块
i2c = SoftI2C(sda=Pin(42), scl=Pin(40))S1 = PWM(Pin(4), freq=50, duty=0) # Servo1的引脚是0def Servo(servo,angle):S1.duty(int(((angle+90)*2/180+0.5)/20*1023))#WIFI连接函数
def WIFI_Connect():WIFI_LED=Pin(46, Pin.OUT) #初始化WIFI指示灯wlan = network.WLAN(network.STA_IF) #STA模式wlan.active(True)                   #激活接口start_time=time.time()              #记录时间做超时判断if not wlan.isconnected():print('Connecting to network...')wlan.connect('FM-674614', '12345678') #输入WIFI账号密码while not wlan.isconnected():#LED闪烁提示WIFI_LED.value(1)time.sleep_ms(300)WIFI_LED.value(0)time.sleep_ms(300)#超时判断,15秒没连接成功判定为超时if time.time()-start_time > 15 :print('WIFI Connected Timeout!')breakif wlan.isconnected():#LED点亮WIFI_LED.value(1)#串口打印信息print('network information:', wlan.ifconfig())return Trueelse:return False#判断WIFI是否连接成功
if WIFI_Connect():#创建socket连接TCP类似,连接成功后发送“Hello 01Studio!”给服务器。s=usocket.socket()addr=('192.168.0.101',10000) #服务器IP和端口s.connect(addr)s.send('Hello 01Studio!')while True:data = s.recv(128)  # 单次最多接收 128 字节if data == b'':passelse:  # 打印接收到的信息为字节,可以通过 decode('utf-8')转成字符串print(data)number = int(data.decode('utf-8'))  # 将接收到的字节数据转换为整数s.send(b'I got: ' + data)  # 发送确认收到的信息给服务器Servo(S1, number)  # 调用 Servo() 函数并传递整数值time.sleep_ms(300)注意,要改成相同的路由WIFI密码,服务器的IP是本机电脑的IP(用ipconfig可以查到)。然后端口写 10000(0-65535 都可以)。
 WIFI  连接成功后返回  True ,否则返回 False 。程序在返回连接成功后建了  Socket  连接,连接成功发送‘ Hello 01Studio!’信息到服务器。另外  RTOS  定时器设定了了每  300ms  处理从服务器接  
  收到的数据。将接收到数据通过串口打印和发送给服务器。 
  
中间的IP地址是电脑本机的IP地址,端口号注意与thonny里面绑定给ESP32的端口号要一致。

成功如下:

 选中后我们在发送框输入信息“ Hi ”,点击发送,可以看到开发板的  REPL  打 235  
  印出来信息  Hi 。为字节数据。另外由于程序将收到的信息发回给服务器,所以在  
  网络调试助手中也接收到开发板返回的信息: I got:Hi 。 
  
你发送数字,舵机可以转动。