应用层简单实现udp / tcp网络通信

2024-09-07 20:04

本文主要是介绍应用层简单实现udp / tcp网络通信,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、常见网络接口总结

1、创建 socket 文件描述符 (TCP/UDP, 客户端 + 服务器)

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

domain:AF_INET:网络通信,AF_LOCAL:本地通信

type:UDP:SOCK_DGRAM,TCP:SOCK_STREAM

protocol:协议编号一开始设0

返回值:文件描述符,Linux下一切皆文件

2、绑定端口号 (TCP/UDP, 服务器)

int bind(int socket, const struct sockaddr *address, socklen_t address_len);

socket:创建socket文件描述符的返回值

address:输入型参数,传递一个带有服务器信息(IP, 端口号, AF_INET)的结构体对象,具体详见http://t.csdnimg.cn/aY2O0

address_len:结构体 address 大小

3、开始监听socket (TCP, 服务器)

int listen(int socket, int backlog);

将套接字设为监听状态,随时准备别人链接服务器

backlog:表示大小,设为8

4、接收请求 (TCP, 服务器)

int accept(int socket, struct sockaddr* address, socklen_t* address_len);

服务器监听状态之后,接受客户端链接

address:输出型参数,得到客户端的信息

address_len:输出型参数,address大小

返回值:文件描述符

对比 accept 函数返回值和 socket

就好比一家餐馆,socket 是门口拉客的,一般叫 listen_sockfd 监听套接字。返回值是客人进餐馆接待客人的服务员,为用户提供网络IO的服务套接字。所以之后真正开始服务的函数用的是accept函数返回值

5、建立连接 (TCP, 客户端)

int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);

客户端在有需要时进行链接服务器

address:输入型参数,传入带有服务器信息的结构体,表示我要链接你这个服务器。

address_len:结构体 address 大小

6、把字符串ip转化成网络序列ip

in_addr_t inet_addr(const char *cp);(不安全)
int inet_pton(int af, const char *src, void *dst);(推荐)

dst:直接填 struct sockaddr_in 中的 sin_addr

7、把网络序列ip转化成字符串ip

char *inet_ntoa(struct in_addr in);(不安全)内部只维护一块静态空间存储IP,会导致覆盖问题

const char *inet_ntop(int af, const void *src, char *dst, socklen_t size);(推荐)
af:AF_INET:网络通信,AF_LOCAL:本地通信

src:4字节网络序列IP

dst:传入自己定义的缓冲区存放ip

size:缓冲区大小

8、从网络中收数据

ssize_t recv(int sockfd, void *buf, size_t len, int flags);

用于tcp读取数据

flag:设置0,阻塞读取

由于tcp有链接,且sockfd本质是文件描述符,可以用read函数

ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags, struct sockaddr *src_addr, socklen_t *addrlen);

用于udp读取数据

src_addr:输出型参数,客户端信息

addrlen:输出型参数,src_addr大小

9、在网络中发数据

ssize_t send(int sockfd, const void *buf, size_t len, int flags);

用于tcp发数据

flag:设置0,阻塞发送

由于tcp有链接,且sockfd本质是文件描述符,可以用write函数

ssize_t sendto(int sockfd, const void *buf, size_t len, int flags, const struct sockaddr *dest_addr, socklen_t addrlen);

用于udp发数据

dest_addr:输入型参数,服务器信息

addrlen:dest_addr大小

10、库函数做网络字节序和主机字节序的转换接口

二、udp_echo_server

1、服务器初始化服务

    void InitServer(){// 1.创建socket文件描述符 一般是3_sockfd = ::socket(AF_INET, SOCK_DGRAM, 0);if(_sockfd < 0){LOG(FATAL, "socket error\n");exit(SOCKET_ERROR); }LOG(DEBUG, "socket create success, _sockfd = %d\n", _sockfd);// 2.先填充跨网络地址信息struct sockaddr_in local;memset(&local, 0, sizeof local);// 通信方式local.sin_family = AF_INET;// 主机转成网络序列local.sin_port = htons(_port);// 要4字节ip 要网络序列ip// local.sin_addr.s_addr = inet_addr(_localip.c_str());// 服务器端进行任意IP绑定local.sin_addr.s_addr = INADDR_ANY;// 3.绑定端口号 把地址信息绑定进套接字int n = ::bind(_sockfd, (struct sockaddr*)&local, sizeof local);if(n < 0){LOG(FATAL, "bind error\n");exit(BIND_ERROR); }LOG(DEBUG, "bind success\n");}

(1)创建sockfd文件描述符

(2)为了绑定函数,初始化 struct socket_in server,并填入服务器信息

注意:服务器一定要绑定 INADDR_ANY 任意IP,这样才能接收到全部IP地址主机的数据

(3)绑定端口号以及一系列服务器信息

2、服务器启动服务

    // 绑定成功就开始服务void Start(){_isrunning = true;char inbuffer[1024];// 死循环while(_isrunning){struct sockaddr_in peer;socklen_t len = sizeof peer;// 服务器收客户端消息ssize_t n = recvfrom(_sockfd, inbuffer, sizeof inbuffer - 1, 0, (struct sockaddr*)&peer, &len);if(n > 0){// 获取哪一个客户端的端口号和IP地址InetAddr addr(peer);inbuffer[n] = 0;cout << "[" << addr.Ip() << ":" << addr.Port() << "]#" << inbuffer << endl;string echo_string = "[udp_server echo] #";echo_string += inbuffer;// 服务器发给客户端消息sendto(_sockfd, echo_string.c_str(), echo_string.size(), 0, (struct sockaddr*)&peer, len);}}}

至少服务器要一直死循环工作

3、客户端初始化

    // 1.获取服务器IP和端口号string server_ip = argv[1];uint16_t server_port = stoi(argv[2]);// 2.绑定服务器// 绑定// 客户端端口号一般不让用户自己设定,让客户端OS随机选择端口// 客户端一定需要绑定自己的IP地址和端口号,但不是显示绑定// 客户端在首次向服务器发送数据时,OS自动给客户端绑定IP和端口号struct sockaddr_in server;memset(&server, 0, sizeof server);server.sin_family = AF_INET;server.sin_port = htons(server_port);server.sin_addr.s_addr = inet_addr(server_ip.c_str());int sockfd = ::socket(AF_INET, SOCK_DGRAM, 0);if(sockfd < 0){cerr << "create socket error" << endl;exit(1);}

(1)获取要连接的服务器信息

(2)初始化 struct socket_in server,并填入服务器信息

(3)创建套接字

(4)重点

客户端不需要显示绑定端口号,也不能自己绑定。但是不代表客户端不要绑定,在第一次客户端给服务器发数据时让客户端的OS随机绑定。

为什么不让用户自己绑定?
服务器可以自己绑定是因为服务器一般只运行一个进程服务,端口号不会冲突,但是客户端不一样,用户的操作系统上会有多个客户端,如果让用户自己设置就很可能端口号冲突,导致用户不能同时启动两个端口号一样的客户端。

4、客户端链接服务器

    while(1){//发送的数据string line;cout << "Please Enter# : ";getline(cin, line);int n = sendto(sockfd, line.c_str(), line.size(), 0, (struct sockaddr*)&server, sizeof server);if(n > 0){struct sockaddr_in temp;socklen_t len = sizeof(temp);char buffer[1024];int m = recvfrom(sockfd, buffer, sizeof buffer - 1, 0, (struct sockaddr*)&temp, &len);if(m > 0){buffer[m] = 0;cout << buffer << endl;}elsebreak;}elsebreak;}

三、tcp_echo_server

1、服务器初始化服务

    void InitServer(){// 1.创建socket_listensockfd = socket(AF_INET, SOCK_STREAM, 0);if(_listensockfd < 0){LOG(FATAL, "socket create error\n");exit(SOCKET_ERROR);}LOG(INFO, "socket create success, sockfd: %d\n", _listensockfd);struct sockaddr_in local;memset(&local, 0, sizeof local);local.sin_family = AF_INET;local.sin_port = htons(_port);local.sin_addr.s_addr = INADDR_ANY;// 2.绑定if(::bind(_listensockfd, (struct sockaddr*)&local, sizeof local) < 0){LOG(FATAL, "socket bind error\n");exit(BIND_ERROR);}LOG(INFO, "bind success\n");// 3.因为tcp是面向连接的,tcp需要不断能获取链接状态,设置为listen状态if(::listen(_listensockfd, gblcklog) < 0){LOG(FATAL, "listen error\n");exit(LISTEN_ERROR);}LOG(INFO, "listen success\n");}

(1)创建sockfd文件描述符

(2)为了绑定函数,初始化 struct socket_in server,并填入服务器信息

注意:服务器一定要绑定 INADDR_ANY 任意IP,这样才能接收到全部IP地址主机的数据

(3)绑定端口号以及一系列服务器信息

(4)开始监听

2、服务器启动服务

    // 内部类 为了拿到新线程要用的sockfd和this指针调用Service函数class ThreadData{public:int _sockfd;TcpServer* _self;InetAddr _addr;public:ThreadData(int sockfd, TcpServer* p, const InetAddr &addr):_sockfd(sockfd),_self(p),_addr(addr){}};void Loop(){// 推荐做法 忽略子进程退出信息// signal(SIGCHLD, SIG_IGN);_isrunning = true;struct sockaddr_in client;socklen_t len = sizeof client;while(_isrunning){// 4.获取链接int sockfd = accept(_listensockfd, (struct sockaddr*)&client, &len);// listensockfd获取失败if(sockfd < 0){LOG(WARNING, "accept error\n");continue;}// 获取成功InetAddr addr(client);LOG(INFO, "get a new link, client info: %s\n", addr.AddStr().c_str());// 版本一 无法并发访问服务器// Service(sockfd, addr);// 版本二 多进程版并发访问服务器// pid_t id = fork();// if(id == 0)// {//     // 继承下来的父进程的_listensockfd(3)需要在子进程关闭,子进程使用sockfd(4)//     ::close(_listensockfd);//     // 创建孙子进程后,子进程退出,此时孙子进程成为孤儿进程,执行服务,死活只由OS关心//     if(fork() > 0)//     {//         exit(0);//     }//     Service(sockfd, addr);//     exit(0);// }// // 同理父进程也要关闭sockfd(4)因为那是属于子进程的,用_listensockfd(3)// // 必须要释放子进程的sockfd, 防止文件描述符泄漏 // int n = waitpid(id, nullptr, 0);// if(n > 0)// {//     LOG(INFO, "wait chid success\n");// }// 版本三 多线程版并发访问服务器// pthread_t tid;// ThreadData* td = new ThreadData(sockfd, this, addr);// // 新线程内部分离线程// pthread_create(&tid, nullptr, Excute, td);// 版本四 线程池版本task_t t = bind(&TcpServer::Service, this, sockfd, addr);ThreadPool<task_t>::GetInstance()->Equeue(t);}_isrunning = false;}// 类里面方法带有this指针,标为静态// 多线程因为共用同一个文件描述符表,所以不能关闭static void* Excute(void* args){pthread_detach(pthread_self());ThreadData* td = static_cast<ThreadData*>(args);td->_self->Service(td->_sockfd, td->_addr);delete(td);return nullptr;}void Service(int sockfd, InetAddr addr){while(1){char inbuffer[1024];ssize_t n = read(sockfd, inbuffer, sizeof inbuffer - 1);if(n > 0){inbuffer[n] = 0;string echo_string = "[server echo] #";echo_string += inbuffer;write(sockfd, echo_string.c_str(), echo_string.size());}else if(n == 0){LOG(INFO, "client %s quit\n", addr.AddStr().c_str());break;}else{LOG(ERROR, "read error: %s\n", addr.AddStr().c_str());break;}}::close(sockfd);}

(1)accpet 函数循环获取新连接,获取成功填入客户端信息

(2)服务端提供服务的四种方式

a、直接调用服务函数

不能并发访问服务器,不使用

b、多进程调用服务函数

创建子进程后,父进程关闭子进程套接字 sockfd,子进程关闭父进程套接字 _listen_sockfd,让父子进程不能相互影响。

signal(SIGCHLD, SIG_IGN); 忽略子进程退出,父进程不用阻塞等待。

也可以在子进程中创建孙子进程,子进程退出,孙子进程变成孤儿进程执行服务函数,函数结束不用父进程管,让OS管。还是推荐忽略子进程退出的方式。

c、多线程调用服务函数

为了线程创建函数中的参数线程调用函数不能有this指针,把调用函数设 static

多线程使用同一张文件描述符表,不能关闭任何套接字

要定义内部类,给线程执行函数传入 sockfd, this指针(调用Service函数), addr

新线程在执行函数中要 pthread_detach(pthread_self()) 进行线程分离

d、线程池调用服务函数

using task_t = function<void()>;

task_t t = bind(&TcpServer::Service, this, sockfd, addr);

ThreadPool<task_t>::GetInstance()->Equeue(t);

3、客户端初始化

    if (argc != 3){cerr << "Usage: " << argv[0] << " server_ip server_port" << endl;exit(0);}string server_ip = argv[1];uint16_t server_port = stoi(argv[2]);// 1.创建socketint sockfd = ::socket(AF_INET, SOCK_STREAM, 0);if(sockfd < 0){cerr << "create socket error" << endl;exit(1);}// 注意:客户端不需要显示绑定,但一定要有自己的ip和port,OS会自动绑定sockfd, 用自己的ip和随机端口号// 向服务器发送链接(第一次发送链接时,OS自动绑定)struct sockaddr_in server;server.sin_family = AF_INET;server.sin_port = htons(server_port);::inet_pton(AF_INET, server_ip.c_str(), &server.sin_addr);

(1)获取要连接的服务器信息

(2)初始化 struct socket_in server,并填入服务器信息

(3)创建套接字

4、客户端链接服务器

    int n = ::connect(sockfd, (struct sockaddr*)&server, sizeof server);if(n < 0){cerr << "connect error" << endl;exit(2);}while(1){string message;cout << "Enter #";getline(cin, message);write(sockfd, message.c_str(), message.size());char echo_buffer[1024];n = read(sockfd, echo_buffer, sizeof echo_buffer);if(n > 0){echo_buffer[n] = 0;cout << echo_buffer << endl;}else{break; }}

这篇关于应用层简单实现udp / tcp网络通信的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1146022

相关文章

SpringBoot集成redisson实现延时队列教程

《SpringBoot集成redisson实现延时队列教程》文章介绍了使用Redisson实现延迟队列的完整步骤,包括依赖导入、Redis配置、工具类封装、业务枚举定义、执行器实现、Bean创建、消费... 目录1、先给项目导入Redisson依赖2、配置redis3、创建 RedissonConfig 配

Python的Darts库实现时间序列预测

《Python的Darts库实现时间序列预测》Darts一个集统计、机器学习与深度学习模型于一体的Python时间序列预测库,本文主要介绍了Python的Darts库实现时间序列预测,感兴趣的可以了解... 目录目录一、什么是 Darts?二、安装与基本配置安装 Darts导入基础模块三、时间序列数据结构与

Python使用FastAPI实现大文件分片上传与断点续传功能

《Python使用FastAPI实现大文件分片上传与断点续传功能》大文件直传常遇到超时、网络抖动失败、失败后只能重传的问题,分片上传+断点续传可以把大文件拆成若干小块逐个上传,并在中断后从已完成分片继... 目录一、接口设计二、服务端实现(FastAPI)2.1 运行环境2.2 目录结构建议2.3 serv

C#实现千万数据秒级导入的代码

《C#实现千万数据秒级导入的代码》在实际开发中excel导入很常见,现代社会中很容易遇到大数据处理业务,所以本文我就给大家分享一下千万数据秒级导入怎么实现,文中有详细的代码示例供大家参考,需要的朋友可... 目录前言一、数据存储二、处理逻辑优化前代码处理逻辑优化后的代码总结前言在实际开发中excel导入很

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

Nginx部署HTTP/3的实现步骤

《Nginx部署HTTP/3的实现步骤》本文介绍了在Nginx中部署HTTP/3的详细步骤,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录前提条件第一步:安装必要的依赖库第二步:获取并构建 BoringSSL第三步:获取 Nginx

MyBatis Plus实现时间字段自动填充的完整方案

《MyBatisPlus实现时间字段自动填充的完整方案》在日常开发中,我们经常需要记录数据的创建时间和更新时间,传统的做法是在每次插入或更新操作时手动设置这些时间字段,这种方式不仅繁琐,还容易遗漏,... 目录前言解决目标技术栈实现步骤1. 实体类注解配置2. 创建元数据处理器3. 服务层代码优化填充机制详

Python实现Excel批量样式修改器(附完整代码)

《Python实现Excel批量样式修改器(附完整代码)》这篇文章主要为大家详细介绍了如何使用Python实现一个Excel批量样式修改器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一... 目录前言功能特性核心功能界面特性系统要求安装说明使用指南基本操作流程高级功能技术实现核心技术栈关键函

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima

SpringBoot全局域名替换的实现

《SpringBoot全局域名替换的实现》本文主要介绍了SpringBoot全局域名替换的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录 项目结构⚙️ 配置文件application.yml️ 配置类AppProperties.Ja