你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

C语言、嵌入式应用——与TCP服务端程序进行通信

[复制链接]
gaosmile 发布时间:2020-6-12 17:06

本篇笔记我们再来一起回顾一下socket相关的知识:我们的开发板作为TCP客户端,与TCP服务端程序进行通信。

准备相关工程
  • 硬件:小熊派开发板。
  • 软件:STM32+RT-Thread
  • 开发工具:RT-Thread Studio V1.1.0。; T* y7 I! e, {

实验前提是我们的开发板与我们的PC所处的网络环境在同一网段内。

我们的开发板联网模块时ESP8266。这里需要使用RTT的at_device软件包,这在之前的笔记中已经有介绍:[color=var(--weui-LINK)]【RT-Thread笔记】onenet软件包的使用

微信图片_20200612164753.jpg
微信图片_20200612164944.jpg
RT-Thread的网络框架

在编写代码之前有必要先了解一下RT-Thread的网络框架结构(图片来源:RT-Thread官网):

微信图片_20200612164949.jpg


' n4 N) X; T* a  P% \

从下往上看:

第 1 层:与硬件相关的一些网络模块,这里我们用的是ESP8266。

第 2~4 层:一些中间层。本次实验中我们可以不用深究,我们把这几层看做一个黑盒子,先不用管里面的实现。有精力的朋友可以去研究,初学朋友暂时先别去碰,碰就是劝退。。。不过也可以稍微了解一些这几层是什么。

第 2 层是协议栈层。这些是一些轻量型的、用于嵌入式中的TCP/IP 协议栈 。

第 3 层是网卡层。通过 netdev 网卡层用户可以统一管理各个网卡信息和网络连接状态,并且可以使用统一的网卡调试命令接口。

第 4 层是SAL 套接字抽象层。通过它 RT-Thread 系统能够适配下层不同的网络协议栈,并提供给上层统一的网络编程接口,方便不同协议栈的接入。

第 5 层应用层标准socket接口。其提供一套标准 BSD Socket API。所谓标准就是我们在RT-Thread应用编程中用的网络接口与在PC上进行网络编程所用的接口函数是一样的,如:

微信图片_20200612165044.jpg


# m2 P  s: ]6 E. Q- {

有了这样的一套标准 BSD Socket API,我们的程序就可以在 PC 上编写、调试:

微信图片_20200612165048.gif


, Y/ R/ z! t. j5 s( l4 q

然后再移植相关代码到 RT-Thread 操作系统上,这给我们提供了很大的便利。

其中,第4层和第5层在在代码中是用宏来关联起来的:

微信图片_20200612165300.jpg


2 D) Y0 G& l- P% N+ |1 [' X

更多的关于RT-Thread的网络框架介绍可以查看官网文档:

http://www.rt-thread.org/document/site/programming-manual/sal/sal/#

下面开始编写测试代码,首先我们需要清楚一个TCP客户端-服务端模型:

微信图片_20200612165306.jpg
编写代码(1)编写TCP客户端代码(开发板代码)

我们这里编写的客户端测试代码就是按照上面那个图来一步一步的编写的:

1、创建一个socket

2、连接服务端

3、发送数据

4、阻塞等待接收数据

5、关闭连接

①创建一个socket

用到的接口:

int socket(int domain, int type, int protocol);

我们创建socket相关的代码如下:

/* 创建一个socket,类型是SOCKET_STREAM, TCP类型 */
3 \! q/ k' s$ O, l* B3 u" w5 {4 t: Sif ((sock_fd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1)7 }: E; D5 c+ t5 N( D' S$ E2 B
{
; P' T' A4 K# P7 q3 L0 c6 T. g    /* 创建socket失败 */
. P9 H7 z: J% m/ w    rt_kprintf("Socket error\n");
4 o- i8 I! J/ _# t' R    return -1;
% c3 ~2 A8 C( O}
, [) y8 d5 M5 n% K: B9 W6 U  t
微信图片_20200612165324.jpg

domain / 协议族类型:

  • AF_INET:IPv4
  • AF_INET6:IPv69 U0 X. L5 c( S* e, r

type / 协议类型:

  • SOCK_STREAM:流套接字
  • SOCK_DGRAM:数据报套接字
  • SOCK_RAW:原始套接字
    0 W. T  S# X1 j: A

protocol / 传输协议

  • IPPROTO_TCP
  • IPPROTO_UDP
  • ......$ k3 d) }, V' V+ P- k( s
②连接服务端

用到的接口:

int connect(int s, const struct sockaddr *name, socklen_t namelen);

微信图片_20200612165333.jpg

我们连接服务端相关的代码如下:

左右滑动查看全部代码>>>

/* 从终端获取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
/* 从终端获取端口并转为无符号数据 */
* l2 @! z% A' Z4 Aport = strtoul(argv[2], 0, 10);% `) \0 f7 u4 a" R0 \

: f- ^" k' R3 c( r/ z9 @$ y/* 通过函数入口参数url获得host地址(如果是域名,会做域名解析) */
( _8 t3 {  E$ {: v+ B7 g6 Y1 ?* qhost = gethostbyname(url);, J: \/ g: A! R* c

  o- o* A8 j3 D8 c4 x- |/* 初始化预连接的服务端地址 */
# @# z9 U8 M+ O0 Y$ \, rserver_addr.sin_family = AF_INET;
" c# s! p; Y# \) Zserver_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

$ s( u+ H. w, u+ z2 _1 G/* 连接到服务端 */
1 k1 F3 z3 O/ N% ~1 aif (connect(sock_fd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1)
' n; n4 I. A. F{
: a) R% O" Y! d2 m% [" d! U    /* 连接失败 */
5 a0 {9 c3 @! ^1 W% M- y    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
}
: P( M9 j  C3 P& f! h+ c; O$ delse% I6 F6 B( T4 x- u) v6 q; \
{
( p+ u# \  R0 o8 e4 t    /* 连接成功 */: M0 F9 n' A6 `, x5 G
    rt_kprintf(">>>>>>>>>>>>Connect server(%s %d) success!\n", url, port);
) W9 ]" [2 a- r6 M+ s& C1 c}
0 u$ [! Y! R4 g9 y: z' P
③发送数据

用到的接口:

int send(int s, const void *dataptr, size_t size, int flags);

微信图片_20200612165345.jpg

我们发送数据相关的代码如下:

/* 发送数据 */8 P) R1 H4 N8 u) a: n' d, c7 \& X, f
if (send(sock_fd, argv[3], strlen(argv[3]), 0) < 0)% t" e# L6 t* e) B
{
- j) Z2 f, J% Y     /* 发送失败,关闭这个连接 */
; Q- k& R0 w* t1 U/ `+ c     closesocket(sock_fd);( }2 O+ V. i- t" k7 h! R
     rt_kprintf("\nsend error,close the socket.\r\n");
" U) ?, E3 u% f* T% v }
2 j' X7 w6 v4 I( k+ ^1 A, P3 h else
! ]3 S4 b* H" t9 J  a4 y7 R; V {: s1 h; g1 n$ e! Y( U
     /* 发送成功 */
) K6 R/ H2 K, ]7 j) U: B8 n     rt_kprintf(">>>>>>>>>>>>Send data(%s) to server success!\n", argv[3]);( n8 {2 c% q) V5 Z) \8 }% v6 d
}

# V% }4 l/ [. l& g: n  _④接收数据

用到的接口:

int recv(int s, void *mem, size_t len, int flags);

微信图片_20200612165350.jpg

我们接收数据的相关代码如下:

/* 等待服务端发送过来的数据 */
8 X  J2 |" S" H! s; S5 iif (recv(sock_fd, recv_buf, 100, 0) < 0)
0 z$ d! ]1 H8 V+ y* g{7 g+ q  H; h; w8 c  _5 {* P3 K$ x
    /* 接收失败,关闭这个连接 */
# u) R# G; J2 u0 Z4 v3 W6 j% q    closesocket(sock_fd);
0 t% @. _0 T& }% V+ V1 Q    rt_kprintf("\nreceived error,close the socket.\r\n");
3 N7 `/ N* O6 ?, t/ A* _}: y/ W) j0 Z% O& ?. [) V% ^
else
$ t" Y, C  k% f# F0 P" E{; w$ o: i& N8 y" j
    /* 接收成功,打印收到的数据 */- R8 ^' r5 q# f
    rt_kprintf(">>>>>>>>>>>>Recv data from server: %s\n",recv_buf);
0 a5 \) a, d" a  N! R0 e7 U+ F; g}

% C. O* S( @) R3 ?  h: C  m! `⑤关闭连接

用到的接口:

int closesocket(int s);

微信图片_20200612165353.jpg
(2)编写TCP服务端代码(PC机)

这里提供的是Windows环境下的TCP服务端程序代码,编写思路也是按照上面的TCP客户端-服务端模型来的,相关接口就不详细列举了,直接贴代码吧:

左右滑动查看全部代码>>>

/* 程序:Windows环境下的TCP服务端程序
2 R# R7 I# b- [3 q2 }/ C gcc编译命令:gcc tcp_server.c -lwsock32 -o tcp_server.exe3 ]" d# o/ Q$ e: |

. l3 E$ ~9 A- ?8 W 微信公众号:嵌入式大杂烩
5 f  L/ A2 h7 b( l 作者:ZhengN5 a+ \0 i) }% o' k+ R1 A
*/
/ R& J" U9 I' |$ @% I  t  ?
9 Z  d: n/ k& p
#include <stdio.h>
3 ]# g8 _+ F0 u4 ]) b5 I#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
1 D; w2 {$ C% d( y, p& N( ?) a; A: H+ d
5 A$ x0 _; J/ V5 I7 K" g2 lint 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;
6 E6 @5 ~$ W$ M- O* }! b" j3 i! }: E char Buf[BUF_LEN] = {0};
' z  D) j5 W( U3 W3 A2 g% s7 ?) b, p SOCKADDR ClientAddr;
* h$ d5 u* }- Y5 ` SOCKADDR_IN ServerSockAddr;, @% m, N  F/ o3 Z2 S. U- ~+ I
int addr_size = 0, recv_len = 0;# c" B5 S; E" W

* ^- {' g0 H, w! W  X8 x- T /* sock需要 */
! e2 F" i* L; O' X; {9 z WSAStartup(MAKEWORD(2,2),&wd);  
! q8 |% V) e6 C: E3 l
. T8 I+ e3 H  v$ _2 u printf("===============这是一个TCP服务端程序==============\n");* i1 P, z2 \% s: A5 F* W

8 @3 ^6 a5 E* g1 c: G- v5 | /* 创建服务端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
{
/ u7 k+ d  X- g  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地址
5 E. t+ ]. i+ C- y    ServerSockAddr.sin_addr.s_addr = inet_addr("192.168.1.101");// 本机IP地址
% l! @5 `5 d% W4 J& n: ]6 r    ServerSockAddr.sin_port = htons(1314);       // 端口
$ Z, d" S, g/ [# j+ k, {4 x# U 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);
: J# I8 x7 k7 n3 @* Z( j }  `" P+ t+ {) a6 @& q5 h6 x9 N
  
$ y$ S0 [# q: G9 q' [ /* 进入监听状态 */
* L+ ]9 A) `+ w7 P; P! v 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");
: P6 J$ M  d* `' b3 s  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);
; Z* Q) K8 Q. U9 k
% _5 ?% X+ U5 A3 H2 t/ V while (1)( q  C5 M1 q' S* q
{
0 k) z4 x3 n" v; P  /* 监听客户端请求,accept函数返回一个新的套接字,发送和接收都是用这个套接字 */
+ X) V( ]; X3 C  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' }
  }
1 m4 _# H  j1 m! Y6 P; K  _2 L- ~3 o* i) ?( o5 Y" K5 X( M
  /* 接受客户端的返回数据 */
6 R+ \1 ^, P0 {3 |& v  int recv_len = recv(ClientSock, Buf, BUF_LEN, 0);7 U; [, i; R; q: T" ~1 z
  printf("客户端发送过来的数据为:%s\n", Buf);
' o$ E3 B7 z8 }1 f/ V  8 c7 g4 }- I# d- j
  /* 发送数据到客户端 */
3 p' @+ [0 g. ]( r1 b  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
  /* 清空缓冲区 */
2 n3 @7 V2 Z* e7 L  memset(Buf, 0, BUF_LEN);  
0 g+ u/ ?( l+ C* ^" g }
) q1 X/ Q1 C/ H7 x
3 _# g# S) A: C- \# y4 n /*如果有退出循环的条件,这里还需要清除对socket库的使用*/7 N6 l  e, I! ^( i1 H
/* 关闭服务端套接字 */
3 K* H/ Y) t" D //closesocket(ServerSock);
0 {: i- L' W3 Q    /* 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语言编写的客户端、服务端程序进行验证:

微信图片_20200612165402.gif
2、STM32<-->PC

(1)STM32作为客户端,与PC端我们自己编写的服务端程序进行通信。

微信图片_20200612170000.jpg
微信图片_20200612170024.gif


: D8 g) y1 X9 T' C( K9 _% w

tcp_client命令是我们使用MSH_CMD_EXPORT宏导出的命令,如:

MSH_CMD_EXPORT(tcp_client, tcp_client sample);

我们可在终端按下TAB键或者输入help来查看有没有导出成功:

微信图片_20200612170031.jpg


$ x1 C3 l! ?; k7 J5 L

我们的测试命令格式为:

tcp_client URL PORT DATA

其中,URL 参数代表网址或IP地址,这里是局域网内的TCP通信测试,所以这个参数其实就是我们电脑的IP地址,可以在cmd下输入ipconfig命令进行查看:

微信图片_20200612170035.jpg


4 O( o+ Y; H4 q3 O, {

PORT 参数代表端口。这里要输入的是服务端程序绑定的端口号。端口使用16bit进行编号,即其范围为:0~65536。

但 0~1023 的端口一般由系统分配给特定的服务程序,例如 Web 服务的端口号为 80,FTP 服务的端口号为 21等。

我们这里的服务端程序端口号可以设置为1024~65535范围内的随意一个数。但要注意的是我们输入的测试命令中的PORT参数要与服务端程序绑定的端口一样,否则客户端就连接不上服务端:

微信图片_20200612170039.jpg


3 T$ d/ F4 N; h/ \* f5 ]

DATA参数代表我们要发送给服务端的数据。

需要注意的是,我们在进行测试时需要先启动服务端程序。如果服务端程序还未启动就运行我们的客户端程序,就会出现连接失败:

微信图片_20200612170042.jpg

3 W0 \$ j$ D. e  G* A" b7 m" M, w

(2)STM32作为客户端,PC端网络调试助手作为服务端。

微信图片_20200612170045.jpg

. {. W5 |8 \6 N

从这个网络助手中可以看到在收到数据的同时可以显示出客户端的IP及端口号。客户端的端口号是系统随机分配的(范围为:1024~65535):

微信图片_20200612170050.jpg


7 |$ S* Z/ u! _# \9 T  y

所以我们不关心端口号,但是我们可以查看客户端的IP地址。如:

微信图片_20200612170053.jpg


5 f' `, A' O1 u; o/ ?6 B" |. B

除了这个串口调试助手之外,之前也有分享过一个很好用的socket编程调试工具,有兴趣的朋友可移步至:[color=var(--weui-LINK)]《网络调试助手的简单使用》进行查看。

(3)Python实现服务端

服务端程序可以用C、C++、Python等语言来编写,上面我们用的是C语言。这里我们也来过一把Python瘾:

微信图片_20200612170057.jpg

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
# a4 a- Q0 g4 r5 x! b, x#coding=utf-8
# f: |! W$ r& ~; C. X7 h8 Q#创建TCP服务器0 G5 U# X& D) K  {" \) A
from socket import ** q8 `+ v! z- f9 J# d
from time import ctime
+ ?) m+ s8 I, x" W , 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 #我的端口号
' U6 X, z6 O3 ^8 B- ]7 EBUFSIZ=1024
0 N6 @5 B- A& f& jADDR=(HOST,PORT)
6 }/ W8 f6 P# h8 B 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:
' @. M% Q! O- t1 y    print('waiting for connection...')
  @/ Z  o/ g4 o: U. v    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:
5 b. j) J5 i* j4 b# M        stock_codes = tcpCliSock.recv(BUFSIZ).decode() #收到的客户端的数据需要解码(python3特性)
% A- z7 R: a4 ?# e  m1 q# I: ], _        print('stock_codes = ',stock_codes)    #传入参数stock_codes
; c4 u9 v: q6 q        if not stock_codes:
7 |, i1 M( \5 z$ a& r            break& N9 _! N; t. ]6 ~* z
        tcpCliSock.send(('[%s] %s' %(ctime(),stock_codes)).encode())  #发送给客户端的数据需要编码(python3特性)
, p; x( b) c8 A9 ~/ d( _/ k" j        after_close_simulation = tcpCliSock.recv(BUFSIZ).decode() #收到的客户端的数据需要解码(python3特性)
# M* G- U* ]' t# u" J+ P        print('after_close_simulation = ',after_close_simulation)    #传入参数after_close_simulation1 P6 V2 R* S3 H+ l! I; N1 h
        if not after_close_simulation:
5 @. Z# p3 G' n- f% k6 Q# P$ ?- {. u            break
' A1 y$ K6 T6 J3 ~        tcpCliSock.send(('[%s] %s' %(ctime(),after_close_simulation)).encode())  #发送给客户端的数据需要编码(python3特性)
6 B* Z; n% M& K& N + |  c9 R- i; q9 T; Z; o( y
    tcpCliSock.close()
; E" H9 r$ p& ]6 ~  _9 L, KtcpSerSock.close()
6 G/ w( `3 i- [0 g
收藏 评论1 发布时间:2020-6-12 17:06

举报

1个回答
巡山小妖 回答时间:2020-6-13 09:10:47
不错 学习了
关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版