【网络编程开发】4.socket套接字及TCP的实现框架 5.TCP多进程并发

ops/2024/9/24 7:25:41/

4.socket套接字及TCP的实现框架

Socket套接字

Socket套接字是网络编程中用于实现不同计算机之间通信的一个基本构建块

在现代计算机网络中,Socket套接字扮演着至关重要的角色。它们为应用程序提供了一种方式,通过这种方式,程序能够通过网络发送和接收数据包。以下是对socket的相关介绍:

  1. 定义功能
    • Socket,译为“套接字”,是一种软件形式的"插座",使得不同主机之间的进程可以通过网络进行通信。它基本上充当了应用程序、操作系统与网络硬件之间的接口。
  2. 核心作用
    • Socket套接字允许一个程序通过网络将信息传递给另一个程序。这涉及到底层的网络协议操作,如TCP/IP,但Socket API提供了一个更简单的抽象,使得程序员无需深入了解这些复杂协议的细节即可进行网络编程。
  3. 类型应用
    • 流式套接字(SOCK_STREAM)基于TCP协议,提供面向连接的可靠数据传输服务,适用于需要确保数据完整性的应用,如网页浏览。
    • 数据报套接字(SOCK_DGRAM)基于UDP协议,提供无连接的服务,适用于对实时性要求高但可以容忍少量数据丢失的应用,如视频通话。
  4. 工作过程
    • 服务器端创建一个Socket并绑定到一个特定的IP地址和端口上,然后开始监听来自客户端的连接请求。
    • 客户端创建一个Socket,并尝试连接到服务器的IP地址和端口。一旦连接建立,双方就可以通过这个Socket进行数据的发送和接收。
  5. 高级功能
    • 使用Socket,可以实现多种网络应用程序,包括聊天应用、文件传输、远程控制等。在多用户游戏、实时数据处理等方面也广泛应用Socket技术。
    • 在多线程或多进程的环境中,Socket也能有效地处理并发连接,提高系统的响应速度和处理能力。

TCP实现框架

在这里插入图片描述

以下是C语言中常见的Socket API及其功能:

以下是C语言中常见的Socket API及其功能:

  1. socket函数:该函数用于创建一个套接字,并返回一个套接字描述符。

    #include <sys/types.h>
    #include <sys/socket.h>
    int socket(int domain, int type, int protocol);
    /*
    参数:domain:指定协议族,如AF_INET表示IPv4地址。type:指定套接字类型,如SOCK_STREAM表示提供有序、可靠、双向字节流的连接。protocol:通常设置为0,表示系统将选择与指定类型相匹配的默认协议。
    返回值:成功时返回一个非负整数,表示新创建的套接字描述符。失败时返回-1,并设置errno为相应的错误码。
    */
    
  2. bind函数:此函数用于将套接字绑定到一个特定的地址和端口号上。这对于服务器来说尤其重要,因为它们需要在已知的地址上监听来自客户端的连接请求。

    #include <sys/types.h>
    #include <sys/socket.h>
    int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
    /*
    参数:sockfd:表示套接字的描述符。addr:指向特定于域的套接字地址结构的指针,该结构包含了要绑定的地址信息。addrlen:指定地址结构的大小。
    返回值:成功时返回0。失败时返回-1,并设置errno为相应的错误码。
    */
    
  3. listen函数:服务器使用此函数将套接字设置为监听状态,等待客户端的连接请求。可以指定一个队列限制,以控制同时可处理的连接数。

    #include <sys/types.h>
    #include <sys/socket.h>
    int listen(int sockfd, int backlog);
    /*
    参数:sockfd:表示套接字的描述符。backlog:指定在未完成连接队列中允许的最大连接数。
    返回值:成功时返回0。失败时返回-1,并设置errno为相应的错误码。
    */
    
  4. accept函数:当有客户端连接请求时,服务器使用此函数接受连接,并返回一个新的套接字,用于与客户端通信。

    #include <sys/types.h>
    #include <sys/socket.h>
    int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
    /*
    参数:sockfd:表示套接字的描述符。addr:指向特定于域的套接字地址结构的指针,该结构用于接收客户端的地址信息。addrlen:指向一个变量的指针,该变量用于接收地址结构的大小。
    返回值:成功时返回一个新的套接字描述符,用于与客户端通信。失败时返回-1,并设置errno为相应的错误码。
    */
    
  5. connect函数:该函数用于建立一个到服务器的连接。对于TCP套接字,这将触发TCP的三次握手过程。

    #include <sys/types.h>
    #include <sys/socket.h>
    int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
    /*
    参数:sockfd:表示套接字的描述符。addr:指向特定于域的套接字地址结构的指针,该结构包含了服务器的地址信息。addrlen:指定地址结构的大小。
    返回值:成功时返回0。失败时返回-1,并设置errno为相应的错误码。
    */
    
  6. send和recv函数:这两个函数分别用于发送和接收数据。它们是TCP通信中常用的数据传输函数。

    #include <sys/types.h>
    #include <sys/socket.h>
    ssize_t send(int sockfd, const void *buf, size_t len, int flags);
    ssize_t recv(int sockfd, void *buf, size_t len, int flags);
    /*
    参数:sockfd:表示套接字的描述符。buf:指向要发送或接收数据的缓冲区的指针。len:指定要发送或接收的数据的长度。flags:可选的标志位,用于控制发送或接收的行为。
    返回值:成功时返回实际发送或接收的字节数。失败时返回-1,并设置errno为相应的错误码。
    */
    
  7. sendto和recvfrom函数:对于UDP通信,不需要建立连接,直接使用sendto和recvfrom函数进行数据报文的发送和接收。

    #include <sys/types.h>
    #include <sys/socket.h>
    ssize_t sendto(int sockfd, const void *buf, size_t len, int flags, const struct sockaddr *dest_addr, socklen_t addrlen);
    ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags, struct sockaddr *src_addr, socklen_t *addrlen);
    /*
    参数:sockfd:表示套接字的描述符。buf:指向要发送或接收数据的缓冲区的指针。len:指定要发送或接收的数据的长度。flags:可选的标志位,用于控制发送或接收的行为。dest_addr:指向目标地址结构的指针,用于UDP发送操作。src_addr:指向源地址结构的指针,用于UDP接收操作。addrlen:指向一个变量的指针,用于UDP接收操作,该变量用于接收源地址结构的大小。
    返回值:成功时返回实际发送或接收的字节数。失败时返回-1,并设置errno为相应的错误码。
    */
    

示例-服务端

TCP_socket.c

#include <stdio.h> // 引入标准输入输出库
#include <sys/socket.h> // 引入套接字库
#include <sys/types.h> // 引入类型定义库
#include <stdlib.h> // 引入标准库
#include <arpa/inet.h> // 引入互联网协议库
#include <unistd.h> // 引入Unix标准库
#include <string.h> #define PORT 5001 // 定义端口号为5001
#define BACKLOG 5 // 定义最大连接数为5int main(int argc, char *argv[]) // 主函数,参数为命令行参数个数和参数列表
{int fd,newfd,ret; // 定义文件描述符变量char buf[BUFSIZ] = {}; //BUFSIZ 8142struct sockaddr_in addr; // 定义IPv4地址结构体if(argc < 3){fprintf(stderr,"%s<addr><PORT>",argv[0]);}// 创建套接字fd = socket(AF_INET, SOCK_STREAM, 0);if(fd < 0){perror("socket"); exit(0); }// 设置地址结构体addr.sin_family = AF_INET; // 使用IPv4协议addr.sin_port = htons(atoi(argv[2])); // 设置端口号,atoi()将字符串转换为整数if(inet_aton(argv[1],&addr.sin_addr)==0){//inet_aton()函数将字符串形式的IP地址转换为网络字节序的二进制形式fprintf(stderr,"Invalid address\n");exit(EXIT_FAILURE);}// 绑定通信结构体if(bind(fd, (struct sockaddr *)&addr, sizeof(addr) ) == -1){perror("bind"); exit(0); }// 设置套接字为监听模式if(listen(fd, BACKLOG) == -1){perror("listen"); exit(0); }// 接受客户端的连接请求,生成新的用于和客户端通信的套接字newfd = accept(fd, NULL, NULL);if(newfd < 0){perror("accept"); exit(0);}/*循环接收*/while(1){memset(buf,0,BUFSIZ);ret = read(newfd,buf,BUFSIZ);if(ret < 0){perror("read");exit(0);}else if(ret == 0)break;elseprintf("buf = %s\n",buf);}close(newfd);close(fd); // 关闭套接字return 0; 
}

示例-客户端

TCP_socket2.c

#include <stdio.h> // 引入标准输入输出库
#include <sys/socket.h> // 引入套接字库
#include <sys/types.h> // 引入类型定义库
#include <stdlib.h> // 引入标准库
#include <arpa/inet.h> // 引入互联网协议库
#include <unistd.h> // 引入Unix标准库
#include <string.h>#define PORT 5001 // 定义端口号为5001
#define BACKLOG 5 // 定义最大连接数为5
#define STR "Hello World!" // 定义要发送的字符串为"Hello World!"int main(int argc, char *argv[]) // 主函数,参数为命令行参数个数和参数列表
{int fd; // 定义文件描述符变量struct sockaddr_in addr; // 定义IPv4地址结构体char buf[BUFSIZ] = {}; //BUFSIZ 8142if(argc < 3){fprintf(stderr,"%s<addr><port>\n",argv[0]);exit(0);}// 创建套接字fd = socket(AF_INET, SOCK_STREAM, 0);if(fd < 0){perror("socket"); // 如果创建失败,打印错误信息exit(0); // 退出程序}// 设置地址结构体addr.sin_family = AF_INET; // 使用IPv4协议addr.sin_port = htons(atoi(argv[2])); // 设置端口号,atoi()将字符串转换为整数if(inet_aton(argv[1],&addr.sin_addr)==0){//inet_aton()函数将字符串形式的IP地址转换为网络字节序的二进制形式fprintf(stderr,"Invalid address\n");exit(EXIT_FAILURE);}    // 向服务端发起连接请求if(connect(fd, (struct sockaddr *)&addr, sizeof(addr) ) == -1){perror("connect"); exit(0); }/*循环发送*/while(1){printf(">");fgets(buf,BUFSIZ,stdin);write(fd,buf,strlen(buf));}// 关闭套接字close(fd);return 0; // 返回0表示程序正常结束
}

在这里插入图片描述

通信成功

如果结束程序,再次运行服务端程序,会出现报错如下(地址已经被使用)

在这里插入图片描述

解决办法:

在程序中加入如下程序段:

/*地址快速重用*/
int flag=1,len= sizeof (int); 
if ( setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &flag, len) == -1) { perror("setsockopt"); exit(1); 
} 

5.TCP多进程并发

示例-服务端

my_sever.c

#include <stdio.h> // 引入标准输入输出库
#include <sys/socket.h> // 引入套接字库
#include <sys/types.h> // 引入类型定义库
#include <stdlib.h> // 引入标准库
#include <arpa/inet.h> // 引入网络地址库
#include <unistd.h> // 引入Unix系统调用库
#include <string.h> // 引入字符串处理库
#include <strings.h> // 引入字符串操作库
#include <signal.h> // 引入信号处理库
#include <sys/wait.h> // 引入进程等待库#define BACKLOG 5 // 定义最大连接数为5void ClinetHandle(int newfd); // 客户端处理函数/*信号处理函数,防止出现僵尸进程*/
void SigHandle(int sig){ if(sig == SIGCHLD){ // 如果接收到SIGCHLD信号printf("client exited\n"); // 打印客户端退出信息wait(NULL); // 等待子进程结束}
}/*主函数*/
int main(int argc, char *argv[])
{int fd, newfd; // 定义文件描述符变量struct sockaddr_in addr, clint_addr; // 定义套接字地址结构体socklen_t addrlen = sizeof(clint_addr); // 定义地址长度变量#if 0struct sigaction act; // 定义信号处理结构体act.sa_handler = SigHandle; // 设置信号处理函数act.sa_flags = SA_RESTART; // 设置信号处理标志sigemptyset(&act.sa_mask); // 清空信号屏蔽集sigaction(SIGCHLD, &act, NULL); // 注册信号处理函数
#elsesignal(SIGCHLD, SigHandle); // 注册信号处理函数
#endifpid_t pid; // 定义进程ID变量if(argc < 3){ // 如果参数个数小于3fprintf(stderr, "%s<addr><port>\n", argv[0]);exit(0); }/*创建套接字*/fd = socket(AF_INET, SOCK_STREAM, 0); // 创建TCP套接字if(fd < 0){ perror("socket"); }addr.sin_family = AF_INET; // 设置地址族为IPv4addr.sin_port = htons( atoi(argv[2]) ); // 设置端口号if ( inet_aton(argv[1], &addr.sin_addr) == 0) { // 如果转换IP地址失败fprintf(stderr, "Invalid address\n"); exit(EXIT_FAILURE); }/*地址快速重用*/int flag=1,len= sizeof (int); if ( setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &flag, len) == -1) { // 如果设置失败perror("setsockopt"); exit(1); } /*绑定通信结构体*/if(bind(fd, (struct sockaddr *)&addr, sizeof(addr) ) == -1){ // 如果绑定失败perror("bind"); exit(0); }/*设置套接字为监听模式*/if(listen(fd, BACKLOG) == -1){ perror("listen"); exit(0); }while(1){ // 循环监听连接请求/*接受客户端的连接请求,生成新的用于和客户端通信的套接字*/newfd = accept(fd, (struct sockaddr *)&clint_addr, &addrlen); // 接受连接请求if(newfd < 0){ perror("accept"); exit(0); }/*打印客户端地址和端口号*/ // inet_ntoa()将网络字节序转换为点分十进制IP地址格式的字符串;// ntohs()将一个16位数从网络字节顺序转换为主机字节顺序printf("addr:%s port:%d\n", inet_ntoa(clint_addr.sin_addr), ntohs(clint_addr.sin_port) ); if( (pid = fork() ) < 0){ perror("fork"); exit(0); }else if(pid == 0){ // 如果当前进程是子进程close(fd); // 关闭父进程的文件描述符ClinetHandle(newfd); // 处理客户端请求exit(0); }elseclose(newfd); // 关闭子进程的文件描述符}close(fd); // 关闭服务器套接字return 0; 
}/*客户端处理函数*/
void ClinetHandle(int newfd){int ret; // 定义返回值变量char buf[BUFSIZ] = {}; // 定义缓冲区数组并初始化为0while(1){ // 循环读取客户端数据//memset(buf, 0, BUFSIZ); // 清空缓冲区数组bzero(buf, BUFSIZ); // 使用bzero函数清空缓冲区数组ret = read(newfd, buf, BUFSIZ); // 从客户端读取数据if(ret < 0) {perror("read"); exit(0); }else if(ret == 0) // 如果读取到EOF标志break; // 跳出循环elseprintf("buf = %s\n", buf); // 打印读取到的数据}close(newfd); // 关闭与客户端的连接
}

示例-客户端

my_client.c

#include <stdio.h> // 引入标准输入输出库
#include <sys/socket.h> // 引入套接字库
#include <sys/types.h> // 引入类型定义库
#include <stdlib.h> // 引入标准库
#include <arpa/inet.h> // 引入互联网协议族库
#include <unistd.h> // 引入Unix标准库
#include <string.h> // 引入字符串处理库#define BACKLOG 5 // 定义最大连接数为5int main(int argc, char *argv[]) // 主函数,参数为命令行参数个数和参数列表
{int fd; // 定义文件描述符变量struct sockaddr_in addr; // 定义套接字地址结构体char buf[BUFSIZ] = {}; // 定义缓冲区数组并初始化为0if(argc < 3){ // 如果参数个数小于3fprintf(stderr, "%s<addr><port>\n", argv[0]); // 打印错误信息exit(0); // 退出程序}/*创建套接字*/fd = socket(AF_INET, SOCK_STREAM, 0); // 创建TCP套接字if(fd < 0){ perror("socket"); exit(0); }addr.sin_family = AF_INET; // 设置地址族为IPv4addr.sin_port = htons( atoi(argv[2]) ); // 设置端口号if ( inet_aton(argv[1], &addr.sin_addr) == 0) { fprintf(stderr, "Invalid address\n"); exit(EXIT_FAILURE); }/*向服务端发起连接请求*/if(connect(fd, (struct sockaddr *)&addr, sizeof(addr) ) == -1){ // 如果连接失败perror("connect"); exit(0); }while(1){ // 循环读取用户输入的数据printf("Input->"); // 提示用户输入fgets(buf, BUFSIZ, stdin); // 从标准输入读取数据到缓冲区write(fd, buf, strlen(buf) ); // 将缓冲区的数据发送给服务器}close(fd); // 关闭套接字return 0; 
}

在这里插入图片描述


http://www.ppmy.cn/ops/47658.html

相关文章

鸿蒙轻内核M核源码分析系列七 动态内存Dynamic Memory

内存管理模块管理系统的内存资源&#xff0c;它是操作系统的核心模块之一&#xff0c;主要包括内存的初始化、分配以及释放。 在系统运行过程中&#xff0c;内存管理模块通过对内存的申请/释放来管理用户和OS对内存的使用&#xff0c;使内存的利用率和使用效率达到最优&#x…

SOCKS 代理 和 HTTP 代理, WebSocket

SOCKS 代理 和 HTTP 代理 的区别 SOCKS 代理 和 HTTP 代理 都是代理服务器&#xff0c;它们充当客户端和目标服务器之间的中介&#xff0c;但它们的工作方式和应用场景有所不同。 1. SOCKS 代理&#xff1a; 工作原理&#xff1a; SOCKS 代理是一种更底层的代理&#xff0c;…

一维时间序列突变检测方法(小波等,MATLAB R2021B)

信号的突变点检测问题是指在生产实践中&#xff0c;反映各种系统工作状态的信号&#xff0c;可能因为受到不同类型的噪声或外界干扰而发生了信号突变&#xff0c;导致严重失真的信号出现&#xff0c;因此必须探测突变出现的起点和终点。研究目的在于设计出检测方案&#xff0c;…

安全风险 - 检测设备是否为模拟器

在很多安全机构的检测中&#xff0c;关于模拟器的运行环境一般也会做监听处理&#xff0c;有的可能允许执行但是会提示用户&#xff0c;有的可能直接禁止在模拟器上运行我方APP 如何判断当前 app 是运行在Android真机&#xff0c;还是运行在模拟器? 可能做 Framework 的朋友思…

k8s群集调度之 pod亲和 node亲和 标签指定

目录 一 调度约束 1.1K8S的 List-Watch 机制 ⭐⭐⭐⭐⭐ 1.1.1Pod 启动典型创建过程 二、调度过程 2.1Predicate&#xff08;预选策略&#xff09; 常见的算法 2.2priorities&#xff08;优选策略&#xff09;常见的算法 三、k8s将pod调度到指定node的方法 3.1指…

【C#】中托管与非托管对象区别、托管与非托管DLL区别

C 中的托管与非托管的区别_托管程序和非托管程序-CSDN博客 C# 中托管与非托管对象区别 在C#中&#xff0c;托管对象和非托管对象的主要区别在于内存管理和执行环境&#xff1a; 托管对象 (Managed Objects) 内存管理&#xff1a;托管对象的内存由.NET运行时&#xff08;CLR…

植物大战僵尸杂交版2.0.88最新版+防闪退工具V2+修改工具+高清工具

植物大战僵尸杂交版&#xff0c;不仅继承原作的经典玩法&#xff0c;而且引入了全新的植物融合玩法&#xff0c;将各式各样的植物进行巧妙的杂交&#xff0c;孕育出前所未有、功能各异的全新植物。 创新的杂交合成系统 游戏引入了创新的杂交合成系统&#xff0c;让玩家可以将不…

跟着Nature学分析:R语言ggplot2包绘制高端组合图

数据和代码获取&#xff1a;请查看主页个人信息&#xff01;&#xff01;&#xff01; 大家好&#xff0c;今天我将介绍高分杂志组合图绘制&#xff0c;图形来源于Nature的一篇文章&#xff1a; 该图展示了转录组多样性的主成分&#xff08;PCs&#xff09;与基因组&#xff08…