C语言、嵌入式应用:TCP通信实例分析

嵌入式客栈 2020-09-15 00:00

点击上方「嵌入式大杂烩」,选择「置顶公众号」第一时间查看嵌入式笔记!

前言

关于socket的笔记,之前已经有分享过两篇相关的文章:

【socket笔记】TCP、UDP通信总结

【socket应用】基于C语言的天气客户端的实现

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

准备相关工程

  • 硬件:小熊派开发板。
  • 软件:STM32+RT-Thread
  • 开发工具:RT-Thread Studio V1.1.0。

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

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

RT-Thread的网络框架

在编写代码之前有必要先了解一下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的网络框架介绍可以查看官网文档:

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

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

编写代码

(1)编写TCP客户端代码(开发板代码)

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

1、创建一个socket

2、连接服务端

3、发送数据

4、阻塞等待接收数据

5、关闭连接

①创建一个socket

用到的接口:

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

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

/* 创建一个socket,类型是SOCKET_STREAM, TCP类型 */
if ((sock_fd = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)) == -1)
{
    /* 创建socket失败 */
    rt_kprintf("Socket error\n");
    return -1;
}

domain / 协议族类型:

  • AF_INET:IPv4
  • AF_INET6:IPv6

type / 协议类型:

  • SOCK_STREAM:流套接字
  • SOCK_DGRAM:数据报套接字
  • SOCK_RAW:原始套接字

protocol / 传输协议

  • IPPROTO_TCP
  • IPPROTO_UDP
  • ......

②连接服务端

用到的接口:

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

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

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

/* 从终端获取URL */
url = argv[1];

/* 从终端获取端口并转为无符号数据 */
port = strtoul(argv[2], 010);

/* 通过函数入口参数url获得host地址(如果是域名,会做域名解析) */
host = gethostbyname(url);

/* 初始化预连接的服务端地址 */
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(port);
server_addr.sin_addr = *((struct in_addr *)host->h_addr);
rt_memset(&(server_addr.sin_zero), 0sizeof(server_addr.sin_zero));

/* 连接到服务端 */
if (connect(sock_fd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1)
{
    /* 连接失败 */
    rt_kprintf("Connect fail!\n");
    closesocket(sock_fd);
    return -1;
}
else
{
    /* 连接成功 */
    rt_kprintf(">>>>>>>>>>>>Connect server(%s %d) success!\n", url, port);
}

③发送数据

用到的接口:

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

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

 /* 发送数据 */
 if (send(sock_fd, argv[3], strlen(argv[3]), 0) < 0)
 {
     /* 发送失败,关闭这个连接 */
     closesocket(sock_fd);
     rt_kprintf("\nsend error,close the socket.\r\n");
 }
 else
 {
     /* 发送成功 */
     rt_kprintf(">>>>>>>>>>>>Send data(%s) to server success!\n", argv[3]);
 }

④接收数据

用到的接口:

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

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

/* 等待服务端发送过来的数据 */
if (recv(sock_fd, recv_buf, 1000) < 0)
{
    /* 接收失败,关闭这个连接 */
    closesocket(sock_fd);
    rt_kprintf("\nreceived error,close the socket.\r\n");
}
else
{
    /* 接收成功,打印收到的数据 */
    rt_kprintf(">>>>>>>>>>>>Recv data from server: %s\n",recv_buf);
}

⑤关闭连接

用到的接口:

int closesocket(int s);

(2)编写TCP服务端代码(PC机)

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

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

/* 程序:Windows环境下的TCP服务端程序
 gcc编译命令:gcc tcp_server.c -lwsock32 -o tcp_server.exe
 
 微信公众号:嵌入式大杂烩
 作者:ZhengN
*/


#include <stdio.h>
#include <winsock2.h>

#define BUF_LEN  100

int main(void)
{
 WSADATA wd;
 SOCKET ServerSock, ClientSock;
 char Buf[BUF_LEN] = {0};
 SOCKADDR ClientAddr;
 SOCKADDR_IN ServerSockAddr;
 int addr_size = 0, recv_len = 0;
 
 /* sock需要 */
 WSAStartup(MAKEWORD(2,2),&wd);  
 
 printf("===============这是一个TCP服务端程序==============\n");
 
 /* 创建服务端socket */
 if (-1 == (ServerSock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)))
 {
  printf("socket error!\n");
  exit(1);
 }
 
 /* 设置服务端信息 */
    memset(&ServerSockAddr, 0sizeof(ServerSockAddr));   // 给结构体ServerSockAddr清零
    ServerSockAddr.sin_family = AF_INET;        // 使用IPv4地址
    ServerSockAddr.sin_addr.s_addr = inet_addr("192.168.1.101");// 本机IP地址
    ServerSockAddr.sin_port = htons(1314);       // 端口
 
 /* 绑定套接字 */
    if (-1 == bind(ServerSock, (SOCKADDR*)&ServerSockAddr, sizeof(SOCKADDR)))
 {
  printf("bind error!\n");
  exit(1);
 }
  
 /* 进入监听状态 */
 if (-1 == listen(ServerSock, 10))
 {
  printf("listen error!\n");
  exit(1);
 }
 
 addr_size = sizeof(SOCKADDR);

 while (1)
 {
  /* 监听客户端请求,accept函数返回一个新的套接字,发送和接收都是用这个套接字 */
  if (-1 == (ClientSock = accept(ServerSock, (SOCKADDR*)&ClientAddr, &addr_size)))
  {
   printf("socket error!\n");
   exit(1);
  }

  /* 接受客户端的返回数据 */
  int recv_len = recv(ClientSock, Buf, BUF_LEN, 0);
  printf("客户端发送过来的数据为:%s\n", Buf);
  
  /* 发送数据到客户端 */
  send(ClientSock, Buf, recv_len, 0);
  
  /* 关闭客户端套接字 */
  closesocket(ClientSock);
  
  /* 清空缓冲区 */
  memset(Buf, 0, BUF_LEN);  
 }

 /*如果有退出循环的条件,这里还需要清除对socket库的使用*/
 /* 关闭服务端套接字 */
 //closesocket(ServerSock);
    /* WSACleanup();*/

 return 0;
}

验证、分析

1、PC端自验证

我们使用我们自己用C语言编写的客户端、服务端程序进行验证:

2、STM32<-->PC

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


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

MSH_CMD_EXPORT(tcp_client, tcp_client sample);

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


我们的测试命令格式为:

tcp_client URL PORT DATA

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


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

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

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


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

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


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


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


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


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

(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服务器
from socket import *
from time import ctime
 
HOST='192.168.1.101' #这个是我的服务器ip,根据情况改动
PORT=1314 #我的端口号
BUFSIZ=1024
ADDR=(HOST,PORT)
 
tcpSerSock=socket(AF_INET,SOCK_STREAM) #创服务器套接字
tcpSerSock.bind(ADDR) #套接字与地址绑定
tcpSerSock.listen(5)  #监听连接,传入连接请求的最大数,一般为5就可以了
 
while True:
    print('waiting for connection...')
    tcpCliSock,addr =tcpSerSock.accept()
    print('...connected from:',addr)
 
    while True:
        stock_codes = tcpCliSock.recv(BUFSIZ).decode() #收到的客户端的数据需要解码(python3特性)
        print('stock_codes = ',stock_codes)    #传入参数stock_codes
        if not stock_codes:
            break
        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_simulation
        if not after_close_simulation:
            break
        tcpCliSock.send(('[%s] %s' %(ctime(),after_close_simulation)).encode())  #发送给客户端的数据需要编码(python3特性) 
 
    tcpCliSock.close()
tcpSerSock.close()

猜你喜欢

基于RT-Thread的智慧路灯案例实验分享

十年经验工程师为何被裁?

串口打印知多少?

嵌入式百宝箱:第2期
于Linux、C、JSON、Socket的编程实例(附代码)

嵌入式客栈 欢迎关注嵌入式客栈,主要分享嵌入式Linux系统构建、嵌入式linux驱动开发、单片机技术、FPGA开发、信号处理、工业通讯等技术主题。欢迎关注,一起交流,一起进步!
评论
  • 新年伊始,又到了对去年做总结,对今年做展望的时刻 不知道你在2024年初立的Flag都实现了吗? 2025年对自己又有什么新的期待呢? 2024年注定是不平凡的一年, 一年里我测评了50余块开发板, 写出了很多科普文章, 从一个小小的工作室成长为科工公司。 展望2025年, 中国香河英茂科工, 会继续深耕于,具身机器人、飞行器、物联网等方面的研发, 我觉得,要向未来学习未来, 未来是什么? 是掌握在孩子们生活中的发现,和精历, 把最好的技术带给孩子,
    丙丁先生 2025-01-11 11:35 463浏览
  •   在信号处理过程中,由于信号的时域截断会导致频谱扩展泄露现象。那么导致频谱泄露发生的根本原因是什么?又该采取什么样的改善方法。本文以ADC性能指标的测试场景为例,探讨了对ADC的输出结果进行非周期截断所带来的影响及问题总结。 两个点   为了更好的分析或处理信号,实际应用时需要从频域而非时域的角度观察原信号。但物理意义上只能直接获取信号的时域信息,为了得到信号的频域信息需要利用傅里叶变换这个工具计算出原信号的频谱函数。但对于计算机来说实现这种计算需要面对两个问题: 1.
    TIAN301 2025-01-14 14:15 110浏览
  • 在不断发展的电子元件领域,继电器——作为切换电路的关键设备,正在经历前所未有的技术变革。固态继电器(SSR)和机械继电器之间的争论由来已久。然而,从未来发展的角度来看,固态继电器正逐渐占据上风。本文将从耐用性、速度和能效三个方面,全面剖析固态继电器为何更具优势,并探讨其在行业中的应用与发展趋势。1. 耐用性:经久耐用的设计机械继电器:机械继电器依靠物理触点完成电路切换。然而,随着时间的推移,这些触点因电弧、氧化和材料老化而逐渐磨损,导致其使用寿命有限。因此,它们更适合低频或对切换耐久性要求不高的
    腾恩科技-彭工 2025-01-10 16:15 102浏览
  • 01. 什么是过程能力分析?过程能力研究利用生产过程中初始一批产品的数据,预测制造过程是否能够稳定地生产符合规格的产品。可以把它想象成一种预测。通过历史数据的分析,推断未来是否可以依赖该工艺持续生产高质量产品。客户可能会要求将过程能力研究作为生产件批准程序 (PPAP) 的一部分。这是为了确保制造过程能够持续稳定地生产合格的产品。02. 基本概念在定义制造过程时,目标是确保生产的零件符合上下规格限 (USL 和 LSL)。过程能力衡量制造过程能多大程度上稳定地生产符合规格的产品。核心概念很简单:
    优思学院 2025-01-12 15:43 529浏览
  • 随着数字化的不断推进,LED显示屏行业对4K、8K等超高清画质的需求日益提升。与此同时,Mini及Micro LED技术的日益成熟,推动了间距小于1.2 Pitch的Mini、Micro LED显示屏的快速发展。这类显示屏不仅画质卓越,而且尺寸适中,通常在110至1000英寸之间,非常适合应用于电影院、监控中心、大型会议、以及电影拍摄等多种室内场景。鉴于室内LED显示屏与用户距离较近,因此对于噪音控制、体积小型化、冗余备份能力及电气安全性的要求尤为严格。为满足这一市场需求,开关电源技术推出了专为
    晶台光耦 2025-01-13 10:42 507浏览
  • 电动汽车(EV)正在改变交通运输,为传统内燃机提供更清洁、更高效的替代方案。这种转变的核心是电力电子和能源管理方面的创新,而光耦合器在其中发挥着关键作用。这些不起眼的组件可实现可靠的通信、增强安全性并优化电动汽车系统的性能,使其成为正在进行的革命中不可或缺的一部分。光耦合器,也称为光隔离器,是一种使用光传输电信号的设备。通过隔离高压和低压电路,光耦合器可确保安全性、减少干扰并保持信号完整性。这些特性对于电动汽车至关重要,因为精确控制和安全性至关重要。 光耦合器在电动汽车中的作用1.电池
    腾恩科技-彭工 2025-01-10 16:14 82浏览
  • 流量传感器是实现对燃气、废气、生活用水、污水、冷却液、石油等各种流体流量精准计量的关键手段。但随着工业自动化、数字化、智能化与低碳化进程的不断加速,采用传统机械式检测方式的流量传感器已不能满足当代流体计量行业对于测量精度、测量范围、使用寿命与维护成本等方面的精细需求。流量传感器的应用场景(部分)超声波流量传感器,是一种利用超声波技术测量流体流量的新型传感器,其主要通过发射超声波信号并接收反射回来的信号,根据超声波在流体中传播的时间、幅度或相位变化等参数,间接计算流体的流量,具有非侵入式测量、高精
    华普微HOPERF 2025-01-13 14:18 486浏览
  • 数字隔离芯片是现代电气工程师在进行电路设计时所必须考虑的一种电子元件,主要用于保护低压控制电路中敏感电子设备的稳定运行与操作人员的人身安全。其不仅能隔离两个或多个高低压回路之间的电气联系,还能防止漏电流、共模噪声与浪涌等干扰信号的传播,有效增强电路间信号传输的抗干扰能力,同时提升电子系统的电磁兼容性与通信稳定性。容耦隔离芯片的典型应用原理图值得一提的是,在电子电路中引入隔离措施会带来传输延迟、功耗增加、成本增加与尺寸增加等问题,而数字隔离芯片的目标就是尽可能消除这些不利影响,同时满足安全法规的要
    华普微HOPERF 2025-01-15 09:48 83浏览
  • PNT、GNSS、GPS均是卫星定位和导航相关领域中的常见缩写词,他们经常会被用到,且在很多情况下会被等同使用或替换使用。我们会把定位导航功能测试叫做PNT性能测试,也会叫做GNSS性能测试。我们会把定位导航终端叫做GNSS模块,也会叫做GPS模块。但是实际上他们之间是有一些重要的区别。伴随着技术发展与越发深入,我们有必要对这三个词汇做以清晰的区分。一、什么是GPS?GPS是Global Positioning System(全球定位系统)的缩写,它是美国建立的全球卫星定位导航系统,是GNSS概
    德思特测试测量 2025-01-13 15:42 498浏览
  • 随着通信技术的迅速发展,现代通信设备需要更高效、可靠且紧凑的解决方案来应对日益复杂的系统。中国自主研发和制造的国产接口芯片,正逐渐成为通信设备(从5G基站到工业通信模块)中的重要基石。这些芯片凭借卓越性能、成本效益及灵活性,满足了现代通信基础设施的多样化需求。 1. 接口芯片在通信设备中的关键作用接口芯片作为数据交互的桥梁,是通信设备中不可或缺的核心组件。它们在设备内的各种子系统之间实现无缝数据传输,支持高速数据交换、协议转换和信号调节等功能。无论是5G基站中的数据处理,还是物联网网关
    克里雅半导体科技 2025-01-10 16:20 448浏览
  • ARMv8-A是ARM公司为满足新需求而重新设计的一个架构,是近20年来ARM架构变动最大的一次。以下是对ARMv8-A的详细介绍: 1. 背景介绍    ARM公司最初并未涉足PC市场,其产品主要针对功耗敏感的移动设备。     随着技术的发展和市场需求的变化,ARM开始扩展到企业设备、服务器等领域,这要求其架构能够支持更大的内存和更复杂的计算任务。 2. 架构特点    ARMv8-A引入了Execution State(执行状
    丙丁先生 2025-01-12 10:30 471浏览
  • 随着全球向绿色能源转型的加速,对高效、可靠和环保元件的需求从未如此强烈。在这种背景下,国产固态继电器(SSR)在实现太阳能逆变器、风力涡轮机和储能系统等关键技术方面发挥着关键作用。本文探讨了绿色能源系统背景下中国固态继电器行业的前景,并强调了2025年的前景。 1.对绿色能源解决方案日益增长的需求绿色能源系统依靠先进的电源管理技术来最大限度地提高效率并最大限度地减少损失。固态继电器以其耐用性、快速开关速度和抗机械磨损而闻名,正日益成为传统机电继电器的首选。可再生能源(尤其是太阳能和风能
    克里雅半导体科技 2025-01-10 16:18 328浏览
  • 根据Global Info Research(环洋市场咨询)项目团队最新调研,预计2030年全球无人机电池和电源产值达到2834百万美元,2024-2030年期间年复合增长率CAGR为10.1%。 无人机电池是为无人机提供动力并使其飞行的关键。无人机使用的电池类型因无人机的大小和型号而异。一些常见的无人机电池类型包括锂聚合物(LiPo)电池、锂离子电池和镍氢(NiMH)电池。锂聚合物电池是最常用的无人机电池类型,因为其能量密度高、设计轻巧。这些电池以输出功率大、飞行时间长而著称。不过,它们需要
    GIRtina 2025-01-13 10:49 198浏览
  • Snyk 是一家为开发人员提供安全平台的公司,致力于协助他们构建安全的应用程序,并为安全团队提供应对数字世界挑战的工具。以下为 Snyk 如何通过 CircleCI 实现其“交付”使命的案例分析。一、Snyk 的挑战随着客户对安全工具需求的不断增长,Snyk 的开发团队面临多重挑战:加速交付的需求:Snyk 的核心目标是为开发者提供更快、更可靠的安全解决方案,但他们的现有 CI/CD 工具(TravisCI)运行缓慢,无法满足快速开发和部署的要求。扩展能力不足:随着团队规模和代码库的不断扩大,S
    艾体宝IT 2025-01-10 15:52 164浏览
  • 食物浪费已成为全球亟待解决的严峻挑战,并对环境和经济造成了重大影响。最新统计数据显示,全球高达三分之一的粮食在生产过程中损失或被无谓浪费,这不仅导致了资源消耗,还加剧了温室气体排放,并带来了巨大经济损失。全球领先的光学解决方案供应商艾迈斯欧司朗(SIX:AMS)近日宣布,艾迈斯欧司朗基于AS7341多光谱传感器开发的创新应用来解决食物浪费这一全球性难题。其多光谱传感解决方案为农业与食品行业带来深远变革,该技术通过精确判定最佳收获时机,提升质量控制水平,并在整个供应链中有效减少浪费。 在2024
    艾迈斯欧司朗 2025-01-14 18:45 68浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦