
本篇笔记我们再来一起回顾一下socket相关的知识:我们的开发板作为TCP客户端,与TCP服务端程序进行通信。 准备相关工程
实验前提是我们的开发板与我们的PC所处的网络环境在同一网段内。 我们的开发板联网模块时ESP8266。这里需要使用RTT的at_device软件包,这在之前的笔记中已经有介绍:[color=var(--weui-LINK)]【RT-Thread笔记】onenet软件包的使用。 ![]() ![]() 在编写代码之前有必要先了解一下RT-Thread的网络框架结构(图片来源:RT-Thread官网): ![]()
从下往上看: 第 1 层:与硬件相关的一些网络模块,这里我们用的是ESP8266。 第 2~4 层:一些中间层。本次实验中我们可以不用深究,我们把这几层看做一个黑盒子,先不用管里面的实现。有精力的朋友可以去研究,初学朋友暂时先别去碰,碰就是劝退。。。不过也可以稍微了解一些这几层是什么。 第 2 层是协议栈层。这些是一些轻量型的、用于嵌入式中的TCP/IP 协议栈 。 第 3 层是网卡层。通过 netdev 网卡层用户可以统一管理各个网卡信息和网络连接状态,并且可以使用统一的网卡调试命令接口。 第 4 层是SAL 套接字抽象层。通过它 RT-Thread 系统能够适配下层不同的网络协议栈,并提供给上层统一的网络编程接口,方便不同协议栈的接入。 第 5 层应用层标准socket接口。其提供一套标准 BSD Socket API。所谓标准就是我们在RT-Thread应用编程中用的网络接口与在PC上进行网络编程所用的接口函数是一样的,如: ![]()
有了这样的一套标准 BSD Socket API,我们的程序就可以在 PC 上编写、调试: ![]()
然后再移植相关代码到 RT-Thread 操作系统上,这给我们提供了很大的便利。 其中,第4层和第5层在在代码中是用宏来关联起来的: ![]()
更多的关于RT-Thread的网络框架介绍可以查看官网文档:
下面开始编写测试代码,首先我们需要清楚一个TCP客户端-服务端模型: ![]() 我们这里编写的客户端测试代码就是按照上面那个图来一步一步的编写的: ①创建一个socket 用到的接口:
我们创建socket相关的代码如下: /* 创建一个socket,类型是SOCKET_STREAM, TCP类型 */if ((sock_fd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1)7 }: E; D5 c+ t5 N( D' S$ E2 B { /* 创建socket失败 */ rt_kprintf("Socket error\n"); return -1; }, [) y8 d5 M5 n% K: B9 W6 U t ![]() domain / 协议族类型:
type / 协议类型:
protocol / 传输协议
用到的接口:
![]() 我们连接服务端相关的代码如下: 左右滑动查看全部代码>>> /* 从终端获取URL */) D N; \- X' Y9 J+ j- [5 C9 ~url = argv[1];4 o% h( V; P. i5 ? l5 _ - j4 R: R/ A% V! l" ^ y /* 从终端获取端口并转为无符号数据 */ port = strtoul(argv[2], 0, 10);% `) \0 f7 u4 a" R0 \ /* 通过函数入口参数url获得host地址(如果是域名,会做域名解析) */ host = gethostbyname(url);, J: \/ g: A! R* c /* 初始化预连接的服务端地址 */ server_addr.sin_family = AF_INET; server_addr.sin_port = htons(port);$ Q7 R, o7 H( A; {1 h server_addr.sin_addr = *((struct in_addr *)host->h_addr);* b# q1 ^" Z& [4 S. c0 M1 i& ` rt_memset(&(server_addr.sin_zero), 0, sizeof(server_addr.sin_zero));2 N& u2 e, C2 V6 T5 X* g /* 连接到服务端 */ if (connect(sock_fd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1) { /* 连接失败 */ rt_kprintf("Connect fail!\n");" X# q" X' S5 |, a$ i closesocket(sock_fd);1 Z+ ?; ]% z4 C+ _) W2 d- z( { return -1;7 X/ H3 n# o/ {2 D( S6 M } else% I6 F6 B( T4 x- u) v6 q; \ { /* 连接成功 */: M0 F9 n' A6 `, x5 G rt_kprintf(">>>>>>>>>>>>Connect server(%s %d) success!\n", url, port); }0 u$ [! Y! R4 g9 y: z' P ③发送数据 用到的接口:
![]() 我们发送数据相关的代码如下: /* 发送数据 */8 P) R1 H4 N8 u) a: n' d, c7 \& X, fif (send(sock_fd, argv[3], strlen(argv[3]), 0) < 0)% t" e# L6 t* e) B { /* 发送失败,关闭这个连接 */ closesocket(sock_fd);( }2 O+ V. i- t" k7 h! R rt_kprintf("\nsend error,close the socket.\r\n"); } else {: s1 h; g1 n$ e! Y( U /* 发送成功 */ rt_kprintf(">>>>>>>>>>>>Send data(%s) to server success!\n", argv[3]);( n8 {2 c% q) V5 Z) \8 }% v6 d } ④接收数据 用到的接口:
![]() 我们接收数据的相关代码如下: /* 等待服务端发送过来的数据 */if (recv(sock_fd, recv_buf, 100, 0) < 0) {7 g+ q H; h; w8 c _5 {* P3 K$ x /* 接收失败,关闭这个连接 */ closesocket(sock_fd); rt_kprintf("\nreceived error,close the socket.\r\n"); }: y/ W) j0 Z% O& ?. [) V% ^ else {; w$ o: i& N8 y" j /* 接收成功,打印收到的数据 */- R8 ^' r5 q# f rt_kprintf(">>>>>>>>>>>>Recv data from server: %s\n",recv_buf); } ⑤关闭连接 用到的接口:
![]() 这里提供的是Windows环境下的TCP服务端程序代码,编写思路也是按照上面的TCP客户端-服务端模型来的,相关接口就不详细列举了,直接贴代码吧: 左右滑动查看全部代码>>> /* 程序:Windows环境下的TCP服务端程序gcc编译命令:gcc tcp_server.c -lwsock32 -o tcp_server.exe3 ]" d# o/ Q$ e: | 微信公众号:嵌入式大杂烩 作者:ZhengN5 a+ \0 i) }% o' k+ R1 A *// R& J" U9 I' |$ @% I t ? 9 Z d: n/ k& p #include <stdio.h> #include <winsock2.h>2 a7 C; u( K' I+ ?0 O4 T# M- v 5 K- u; S7 ]. z3 a# \4 ]; G- K #define BUF_LEN 100 int main(void)% r9 L, h' u! M! l- g' I/ B {3 k3 a4 v4 `/ } WSADATA wd;; C8 ]( G. V+ | H* N SOCKET ServerSock, ClientSock; char Buf[BUF_LEN] = {0}; SOCKADDR ClientAddr; SOCKADDR_IN ServerSockAddr;, @% m, N F/ o3 Z2 S. U- ~+ I int addr_size = 0, recv_len = 0;# c" B5 S; E" W /* sock需要 */ WSAStartup(MAKEWORD(2,2),&wd); printf("===============这是一个TCP服务端程序==============\n");* i1 P, z2 \% s: A5 F* W /* 创建服务端socket */* w+ a; o$ b% \- R7 e if (-1 == (ServerSock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)))% h( U( u3 M1 N% T+ @7 A { printf("socket error!\n");! ^7 v( k ?3 [% ^/ s8 H; n+ p exit(1);3 z1 ` o' s H2 B* F# D8 g% S }1 t0 C* c# R- |$ Z* Y) q7 n . T" w3 W4 ]- f I /* 设置服务端信息 */3 C. d% c; \3 T; z, _- r8 I# ]+ q memset(&ServerSockAddr, 0, sizeof(ServerSockAddr)); // 给结构体ServerSockAddr清零* d0 o$ M. K5 Z+ w% f7 o ServerSockAddr.sin_family = AF_INET; // 使用IPv4地址 ServerSockAddr.sin_addr.s_addr = inet_addr("192.168.1.101");// 本机IP地址 ServerSockAddr.sin_port = htons(1314); // 端口 8 L( F6 M% v1 }, v; R e) D /* 绑定套接字 */+ ^! o0 O6 e$ j4 {) w if (-1 == bind(ServerSock, (SOCKADDR*)&ServerSockAddr, sizeof(SOCKADDR)))# L5 q9 q, P6 d; ^0 x* o3 H5 l {9 t4 J5 i' ~, ~ printf("bind error!\n"); y1 a. m1 o7 [/ i! k; z) f* T) p' ` exit(1); } `" P+ t+ {) a6 @& q5 h6 x9 N /* 进入监听状态 */ if (-1 == listen(ServerSock, 10)). L/ d( x5 B, T) Y) E: a {! o3 ^/ h) J" p4 ]5 X* O/ |' O printf("listen error!\n"); exit(1);5 w) Z) y5 b6 K- f# I! R' L! W$ K: M }: n6 C0 B. E$ u. N4 y v 3 W, D1 ?6 r! t9 k, S ` addr_size = sizeof(SOCKADDR); while (1)( q C5 M1 q' S* q { /* 监听客户端请求,accept函数返回一个新的套接字,发送和接收都是用这个套接字 */ if (-1 == (ClientSock = accept(ServerSock, (SOCKADDR*)&ClientAddr, &addr_size)))6 W2 p% a9 A8 W5 H* J, y {" _2 k* H! p% ]/ ?5 `0 G/ G printf("socket error!\n");/ R% Q+ O, G* Y; d! O2 M/ [: h exit(1);0 D3 o' W0 o' } } 3 o* i) ?( o5 Y" K5 X( M /* 接受客户端的返回数据 */ int recv_len = recv(ClientSock, Buf, BUF_LEN, 0);7 U; [, i; R; q: T" ~1 z printf("客户端发送过来的数据为:%s\n", Buf); 8 c7 g4 }- I# d- j /* 发送数据到客户端 */ send(ClientSock, Buf, recv_len, 0);3 ]% T: T' [& ` 2 H5 I' }/ D; |3 E$ C /* 关闭客户端套接字 *// p2 C3 J3 b: r closesocket(ClientSock);% j; S, Z0 y2 L: d # h o3 v& V/ w! l /* 清空缓冲区 */ memset(Buf, 0, BUF_LEN); } /*如果有退出循环的条件,这里还需要清除对socket库的使用*/7 N6 l e, I! ^( i1 H /* 关闭服务端套接字 */ //closesocket(ServerSock); /* WSACleanup();*/) E h4 \( I/ F ( b: Q- L" S3 T$ u8 L, e return 0;: j: y3 U2 [5 @( K. ^ }" f: a; h. `8 J( U1 u 验证、分析1、PC端自验证 我们使用我们自己用C语言编写的客户端、服务端程序进行验证: ![]() (1)STM32作为客户端,与PC端我们自己编写的服务端程序进行通信。 ![]() ![]()
tcp_client命令是我们使用MSH_CMD_EXPORT宏导出的命令,如:
我们可在终端按下TAB键或者输入help来查看有没有导出成功: ![]()
我们的测试命令格式为:
其中,URL 参数代表网址或IP地址,这里是局域网内的TCP通信测试,所以这个参数其实就是我们电脑的IP地址,可以在cmd下输入ipconfig命令进行查看: ![]()
PORT 参数代表端口。这里要输入的是服务端程序绑定的端口号。端口使用16bit进行编号,即其范围为:0~65536。 但 0~1023 的端口一般由系统分配给特定的服务程序,例如 Web 服务的端口号为 80,FTP 服务的端口号为 21等。 我们这里的服务端程序端口号可以设置为1024~65535范围内的随意一个数。但要注意的是我们输入的测试命令中的PORT参数要与服务端程序绑定的端口一样,否则客户端就连接不上服务端: ![]()
DATA参数代表我们要发送给服务端的数据。 需要注意的是,我们在进行测试时需要先启动服务端程序。如果服务端程序还未启动就运行我们的客户端程序,就会出现连接失败: ![]() 3 W0 \$ j$ D. e G* A" b7 m" M, w (2)STM32作为客户端,PC端网络调试助手作为服务端。 ![]() . {. W5 |8 \6 N 从这个网络助手中可以看到在收到数据的同时可以显示出客户端的IP及端口号。客户端的端口号是系统随机分配的(范围为:1024~65535): ![]()
所以我们不关心端口号,但是我们可以查看客户端的IP地址。如: ![]()
除了这个串口调试助手之外,之前也有分享过一个很好用的socket编程调试工具,有兴趣的朋友可移步至:[color=var(--weui-LINK)]《网络调试助手的简单使用》进行查看。 (3)Python实现服务端 服务端程序可以用C、C++、Python等语言来编写,上面我们用的是C语言。这里我们也来过一把Python瘾: ![]() Python代码: 以下Python代码来自CSDN博客。博客链接: https://blog.csdn.net/liao392781/article/details/80116600?utm_medium=distribute.pc_relevant.none-task-blog-BlogCommendFromMachineLearnPai2-1.nonecase&depth_1-utm_source=distribute.pc_relevant.none-task-blog-BlogCommendFromMachineLearnPai2-1.nonecase#coding=utf-8 #创建TCP服务器0 G5 U# X& D) K {" \) A from socket import ** q8 `+ v! z- f9 J# d from time import ctime , l. p' G$ h* E: M0 ]& D3 k HOST='192.168.1.101' #这个是我的服务器ip,根据情况改动0 O/ ]9 z' a& y( w- n! f; n PORT=1314 #我的端口号 BUFSIZ=1024 ADDR=(HOST,PORT) 3 u8 y) c5 Z- q: I$ R+ p0 K3 o tcpSerSock=socket(AF_INET,SOCK_STREAM) #创服务器套接字+ O" N- l: _' Y* `8 y$ Y tcpSerSock.bind(ADDR) #套接字与地址绑定8 Y9 X! ]5 h% I5 y' N o tcpSerSock.listen(5) #监听连接,传入连接请求的最大数,一般为5就可以了* n5 W6 {% W7 A' e 8 |) o& p! y1 l# k while True: print('waiting for connection...') tcpCliSock,addr =tcpSerSock.accept()- A8 |+ N: S; M/ W& q8 s print('...connected from:',addr)) \# ?* j9 z$ w. r. N( H& a ( D& }% y% A4 g- q3 F while True: stock_codes = tcpCliSock.recv(BUFSIZ).decode() #收到的客户端的数据需要解码(python3特性) print('stock_codes = ',stock_codes) #传入参数stock_codes if not stock_codes: break& N9 _! N; t. ]6 ~* z tcpCliSock.send(('[%s] %s' %(ctime(),stock_codes)).encode()) #发送给客户端的数据需要编码(python3特性) after_close_simulation = tcpCliSock.recv(BUFSIZ).decode() #收到的客户端的数据需要解码(python3特性) print('after_close_simulation = ',after_close_simulation) #传入参数after_close_simulation1 P6 V2 R* S3 H+ l! I; N1 h if not after_close_simulation: break tcpCliSock.send(('[%s] %s' %(ctime(),after_close_simulation)).encode()) #发送给客户端的数据需要编码(python3特性) + | c9 R- i; q9 T; Z; o( y tcpCliSock.close() tcpSerSock.close()6 G/ w( `3 i- [0 g |
不错 学习了 |
STM32F10xxx 正交编码器接口应用笔记 及源代码
基于STM32定时器ETR信号的应用示例
STM32 生态系统|基于STM32WB的低功耗蓝牙应用(一)
《无刷直流电机控制应用 基于STM8S系列单片机》
STM32定时器触发SPI逐字收发之应用示例
【银杏科技ARM+FPGA双核心应用】STM32H7系列10——ADC
【银杏科技ARM+FPGA双核心应用】STM32H7系列57——MDK_FLM
【STM32图书分享之九】—《STM32F 32位ARM微控制器应用设计与实践》
无刷直流电机控制应用+基于STM8S系列单片机---电子书
STM32 USB的程序,包含固件、驱动和测试用的应用程序