【Linux网络编程】3.sockaddr地址结构、网络套接字函数

2024-05-07 10:28

本文主要是介绍【Linux网络编程】3.sockaddr地址结构、网络套接字函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

sockaddr地址结构

网络套接字函数

socket模型创建流程图

server(服务器端)

client(客户端)

socket

参数domain

参数type

参数protocol

返回值

bind

参数sockfd

参数addr

参数addrlen

返回值

listen

参数sockfd

参数backlog

返回值

accept

参数sockfd

参数addr

参数addrlen

返回值

connect

参数sockfd

参数addr

参数addrlen

返回值

测试代码1

测试结果

测试代码2

测试结果

测试代码3

测试结果

sockaddr地址结构

IP+port:在网络环境中唯一标识一个进程。  

man 7 ip
struct sockaddr {sa_family_t sa_family; 		/* address family, AF_xxx */char sa_data[14];			/* 14 bytes of protocol address */
};struct sockaddr_in {sa_family_t    sin_family;			/* Address family */  	//地址结构类型in_port_t      sin_port;			/* Port number */		//端口号struct in_addr sin_addr;			/* Internet address */	//IP地址
};struct in_addr {						/* Internet address. */uint32_t       s_addr;
};

用法:

struct sockaddr_in addr;
addr.sin_family = IP地址的类型;	//IPv4:AF_INET		IPv6:AF_INET6
addr.sin_port = htons(端口号);//s_addr的获取方式1
int dst;
inet_pton(IP地址的类型, "IP地址", (void *)&dst);
addr.sin_addr.s_addr = dst;//s_addr的获取方式2
addr.sin_addr.s_addr = htonl(INADDR_ANY);	//INADDR_ANY:取出系统中有效的任意IP地址,二进制类型。bind(fd, (struct sockaddr *)&addr, size);

网络套接字函数

socket模型创建流程图

server(服务器端)

  1. socket():创建socket。

  2. bind():绑定服务器地址结构。

  3. listen():设置监听上限。

  4. accept():阻塞监听客户端连接。

  5. read(fd):读socket获取客户端数据。

  6. write(fd)

  7. close()

client(客户端)

  1. socket():创建socket。

  2. connect():与服务器建立连接。

  3. write():写数据到socket。

  4. read():读转换后的数据。

  5. 显示读取结果。

  6. close()。

socket

创建一个套接字。

man socket

参数domain

协议类型。

AF_INET:IPv4

AF_INET6:IPv6

AF_UNIX:本地套接字

参数type

数据传输协议。

SOCK_STREAM:流式协议,TCP传输

SOCK_DGRAM:UDP传输

参数protocol

0:默认值

返回值

成功:新套接字所对应的文件描述符

失败:-1

bind

       给socket绑定一个地址结构(IP+port)。如果不使用bind绑定客户端地址结构, 采用“隐式绑定”。

man bind

参数sockfd

新套接字所对应的文件描述符,socket函数返回值。

参数addr

构造出IP地址加端口号,地址结构。

struct sockaddr_in addr;
addr.sin_family = IP地址的类型;	//IPv4:AF_INET		IPv6:AF_INET6
addr.sin_port = htons(端口号);//s_addr的获取方式1
int dst;
inet_pton(IP地址的类型, "IP地址", (void *)&dst);
addr.sin_addr.s_addr = dst;//s_addr的获取方式2
addr.sin_addr.s_addr = htonl(INADDR_ANY);	//INADDR_ANY:取出系统中有效的任意IP地址,二进制类型。bind(fd, (struct sockaddr *)&addr, size);

参数addrlen

地址结构的大小。

sizeof(addr)

返回值

成功:0

失败:-1

listen

设置同时与服务器建立连接的上限数。同时进行3次握手的客户端数量。

man listen

参数sockfd

新套接字所对应的文件描述符,socket函数返回值。

参数backlog

       上限数值。最大值128。数值越大,服务器丢客户端的概率就越小,连接速率也越快。上千万级数量的客户端同时请求连接单进程的服务器时,效果最明显。

返回值

成功:0

失败:-1

accept

阻塞等待客户端建立连接。

man 2 accept

参数sockfd

新套接字所对应的文件描述符,socket函数返回值。

参数addr

传出参数。成功与服务器建立连接的那个客户端的地址结构(IP+port)。

参数addrlen

传入传出参数,地址长度。

socklen_t clit_addr_len = sizeof(addr);&clit_addr_len

传入:addr的大小。

传出:客户端addr的实际大小。

返回值

成功:能与客户端进行数据通信的socket对应的文件描述。

失败:-1。

connect

使用现有的socket与服务器建立连接。

man 2 connect

参数sockfd

新套接字所对应的文件描述符,socket函数返回值。

参数addr

传入参数。服务器的地址结构。

参数addrlen

服务器的地址结构的大小。

返回值

成功:0

失败:-1

测试代码1

服务器从客户端读取数据,然后将数据回送给客户端。

/*测试1服务器端代码,网络调试助手作客户端*/#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <semaphore.h>
#include <pthread.h>
#include <string.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <ctype.h>int main(int argc, char *argv[])
{int flag;int fd_FWQ;                          //服务器文件描述符int fd_KFD;                          //客户端文件描述符char data[1024];                     //读取的数据int ZiJieShu;                        //字节数struct sockaddr_in DiZhi_JieGou_FWQ; //服务器地址结构struct sockaddr_in DiZhi_JieGou_KHD; //客户端地址结构socklen_t KeHuDuan_DaXiao;           //客户端大小char KHD_IP[1024];                   //客户端IPchar FWQ_IP[1024];                   //服务器IPfd_FWQ = socket(AF_INET, SOCK_STREAM, 0); //创建服务器套接字if (fd_FWQ == -1){perror("创建服务器套接字错误");exit(1);}DiZhi_JieGou_FWQ.sin_family = AF_INET;                //IPv4DiZhi_JieGou_FWQ.sin_port = htons(8080);              //端口号8080DiZhi_JieGou_FWQ.sin_addr.s_addr = htonl(INADDR_ANY); //获取系统中任意有效的IP地址flag = bind(fd_FWQ, (struct sockaddr *)&DiZhi_JieGou_FWQ, sizeof(DiZhi_JieGou_FWQ)); //绑定服务器的地址结构if (flag == -1){perror("绑定服务器地址结构错误");exit(1);}printf("服务器IP:%s,端口号:%d\n",inet_ntop(AF_INET, &DiZhi_JieGou_FWQ.sin_addr.s_addr, FWQ_IP, sizeof(FWQ_IP)),ntohs(DiZhi_JieGou_FWQ.sin_port));flag = listen(fd_FWQ, 128); //设置连接服务器上限数if (flag == -1){perror("设置连接上限数错误");exit(1);}KeHuDuan_DaXiao = sizeof(DiZhi_JieGou_KHD);fd_KFD = accept(fd_FWQ, (struct sockaddr *)&DiZhi_JieGou_KHD, &KeHuDuan_DaXiao); //阻塞监听客户端连接if (fd_KFD == -1){perror("阻塞监听客户端连接错误");exit(1);}printf("客户端IP:%s,端口号:%d\n",inet_ntop(AF_INET, &DiZhi_JieGou_KHD.sin_addr.s_addr, KHD_IP, sizeof(KHD_IP)), //网络转换成十进制本地IPntohs(DiZhi_JieGou_KHD.sin_port));                                             //网络转换成本地端口while (1){ZiJieShu = read(fd_KFD, data, sizeof(data));write(STDOUT_FILENO, data, ZiJieShu); //终端显示write(fd_KFD, data, ZiJieShu);}close(fd_KFD);close(fd_FWQ);return 0;
}

测试结果

服务器IP为0.0.0.0,表示所有地址、不确定地址、任意地址。

虚拟机端:

本机端:  

手机端:  

测试代码2

客户端与服务器的通讯。

/*测试2客户端代码,用网络调试助手作服务器*/#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <semaphore.h>
#include <pthread.h>
#include <string.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <ctype.h>#define FWQ_IP "10.3.22.215" //服务器IPint main(int argc, char *argv[])
{int fd_FWQ; //服务器int flag;struct sockaddr_in FWQ_DiZhi; //服务器地址int ZiFuShu;                  //字符数char data[1024];              //数据fd_FWQ = socket(AF_INET, SOCK_STREAM, 0); //创建服务器套接字if (fd_FWQ == -1){perror("创建服务器套接字错误");exit(1);}FWQ_DiZhi.sin_family = AF_INET;                         //IPv4FWQ_DiZhi.sin_port = htons(10500);                       //端口号8080flag = inet_pton(AF_INET, FWQ_IP, &FWQ_DiZhi.sin_addr); //十进制IP转换网络IPif (flag == -1){perror("十进制IP转换网络IP错误");exit(1);}flag = connect(fd_FWQ, (struct sockaddr *)&FWQ_DiZhi, sizeof(FWQ_DiZhi));if (flag == -1){perror("连接服务器错误");exit(1);}printf("连接服务器完成。\n");while (1){ZiFuShu = read(fd_FWQ, &data, sizeof(data));write(fd_FWQ, "客户端接收到的数据是:", sizeof("客户端接收到的数据是:")); //将数据发回给服务器write(fd_FWQ, data, ZiFuShu);write(STDOUT_FILENO, "服务器发送的数据是:", sizeof("服务器发送的数据是:")); //显示接收到服务器的数据write(STDOUT_FILENO, data, ZiFuShu);sleep(1);}close(fd_FWQ);return 0;
}

测试结果

测试代码3

客户端与服务器实现数据传输。

/*测试3服务器端代码CeShi3_FWQ.c
*/#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <semaphore.h>
#include <pthread.h>
#include <string.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <ctype.h>int main(int argc, char *argv[])
{int flag;int fd_FWQ;                          //服务器文件描述符int fd_KFD;                          //客户端文件描述符char data[1024];                     //读取的数据int ZiJieShu;                        //字节数struct sockaddr_in DiZhi_JieGou_FWQ; //服务器地址结构struct sockaddr_in DiZhi_JieGou_KHD; //客户端地址结构socklen_t KeHuDuan_DaXiao;           //客户端大小char KHD_IP[1024];                   //客户端IPchar FWQ_IP[1024];                   //服务器IPfd_FWQ = socket(AF_INET, SOCK_STREAM, 0); //创建服务器套接字if (fd_FWQ == -1){perror("创建服务器套接字错误");exit(1);}DiZhi_JieGou_FWQ.sin_family = AF_INET;                                               //IPv4DiZhi_JieGou_FWQ.sin_port = htons(8080);                                             //端口号8080DiZhi_JieGou_FWQ.sin_addr.s_addr = htonl(INADDR_ANY);                                //获取系统中任意有效的IP地址flag = bind(fd_FWQ, (struct sockaddr *)&DiZhi_JieGou_FWQ, sizeof(DiZhi_JieGou_FWQ)); //绑定服务器的地址结构if (flag == -1){perror("绑定服务器地址结构错误");exit(1);}printf("服务器IP:%s,端口号:%d\n",inet_ntop(AF_INET, &DiZhi_JieGou_FWQ.sin_addr.s_addr, FWQ_IP, sizeof(FWQ_IP)),ntohs(DiZhi_JieGou_FWQ.sin_port));flag = listen(fd_FWQ, 128); //设置连接服务器上限数if (flag == -1){perror("设置连接上限数错误");exit(1);}KeHuDuan_DaXiao = sizeof(DiZhi_JieGou_KHD);fd_KFD = accept(fd_FWQ, (struct sockaddr *)&DiZhi_JieGou_KHD, &KeHuDuan_DaXiao); //阻塞监听客户端连接if (fd_KFD == -1){perror("阻塞监听客户端连接错误");exit(1);}printf("客户端IP:%s,端口号:%d\n",inet_ntop(AF_INET, &DiZhi_JieGou_KHD.sin_addr.s_addr, KHD_IP, sizeof(KHD_IP)), //网络转换成十进制本地IPntohs(DiZhi_JieGou_KHD.sin_port));                                             //网络转换成本地端口while (1){ZiJieShu = read(fd_KFD, data, sizeof(data));if (ZiJieShu > 0){write(STDOUT_FILENO, "服务器接收到的数据为:", sizeof("服务器接收到的数据为:"));write(STDOUT_FILENO, data, ZiJieShu); //终端显示write(fd_KFD, "你好客户端,我是服务器,接收到的数据为:", sizeof("你好客户端,我是服务器,接收到的数据为:"));write(fd_KFD, data, ZiJieShu);}}close(fd_KFD);close(fd_FWQ);return 0;
}
/*测试3客户端代码CeShi3_KHD.c
*/#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <semaphore.h>
#include <pthread.h>
#include <string.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <ctype.h>#define FWQ_IP "127.0.0.1" //服务器IPint main(int argc, char *argv[])
{int fd_FWQ; //服务器int flag;struct sockaddr_in FWQ_DiZhi; //服务器地址int ZiFuShu;                  //字符数char data[1024];              //数据fd_FWQ = socket(AF_INET, SOCK_STREAM, 0); //创建服务器套接字if (fd_FWQ == -1){perror("创建服务器套接字错误");exit(1);}FWQ_DiZhi.sin_family = AF_INET;                         //IPv4FWQ_DiZhi.sin_port = htons(8080);                       //端口号8080flag = inet_pton(AF_INET, FWQ_IP, &FWQ_DiZhi.sin_addr); //十进制IP转换网络IPif (flag == -1){perror("十进制IP转换网络IP错误");exit(1);}flag = connect(fd_FWQ, (struct sockaddr *)&FWQ_DiZhi, sizeof(FWQ_DiZhi));if (flag == -1){perror("连接服务器错误");exit(1);}printf("连接服务器完成。\n");while (1){write(fd_FWQ, "你好服务器,我是客户端,你好,世界!\n", sizeof("你好服务器,我是客户端,你好,世界!\n")); //将数据发给服务器printf("向服务器发送的数据是:你好服务器,我是客户端,你好,世界!\n");ZiFuShu = read(fd_FWQ, &data, sizeof(data));if (ZiFuShu > 0){write(STDOUT_FILENO, "服务器发送的数据是:", sizeof("服务器发送的数据是:")); //显示接收到服务器的数据write(STDOUT_FILENO, data, ZiFuShu);}sleep(1);}close(fd_FWQ);return 0;
}

测试结果

这篇关于【Linux网络编程】3.sockaddr地址结构、网络套接字函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++中assign函数的使用

《C++中assign函数的使用》在C++标准模板库中,std::list等容器都提供了assign成员函数,它比操作符更灵活,支持多种初始化方式,下面就来介绍一下assign的用法,具有一定的参考价... 目录​1.assign的基本功能​​语法​2. 具体用法示例​​​(1) 填充n个相同值​​(2)

MySql基本查询之表的增删查改+聚合函数案例详解

《MySql基本查询之表的增删查改+聚合函数案例详解》本文详解SQL的CURD操作INSERT用于数据插入(单行/多行及冲突处理),SELECT实现数据检索(列选择、条件过滤、排序分页),UPDATE... 目录一、Create1.1 单行数据 + 全列插入1.2 多行数据 + 指定列插入1.3 插入否则更

Linux进程CPU绑定优化与实践过程

《Linux进程CPU绑定优化与实践过程》Linux支持进程绑定至特定CPU核心,通过sched_setaffinity系统调用和taskset工具实现,优化缓存效率与上下文切换,提升多核计算性能,适... 目录1. 多核处理器及并行计算概念1.1 多核处理器架构概述1.2 并行计算的含义及重要性1.3 并

PostgreSQL中rank()窗口函数实用指南与示例

《PostgreSQL中rank()窗口函数实用指南与示例》在数据分析和数据库管理中,经常需要对数据进行排名操作,PostgreSQL提供了强大的窗口函数rank(),可以方便地对结果集中的行进行排名... 目录一、rank()函数简介二、基础示例:部门内员工薪资排名示例数据排名查询三、高级应用示例1. 每

Linux线程之线程的创建、属性、回收、退出、取消方式

《Linux线程之线程的创建、属性、回收、退出、取消方式》文章总结了线程管理核心知识:线程号唯一、创建方式、属性设置(如分离状态与栈大小)、回收机制(join/detach)、退出方法(返回/pthr... 目录1. 线程号2. 线程的创建3. 线程属性4. 线程的回收5. 线程的退出6. 线程的取消7.

Linux下进程的CPU配置与线程绑定过程

《Linux下进程的CPU配置与线程绑定过程》本文介绍Linux系统中基于进程和线程的CPU配置方法,通过taskset命令和pthread库调整亲和力,将进程/线程绑定到特定CPU核心以优化资源分配... 目录1 基于进程的CPU配置1.1 对CPU亲和力的配置1.2 绑定进程到指定CPU核上运行2 基于

全面掌握 SQL 中的 DATEDIFF函数及用法最佳实践

《全面掌握SQL中的DATEDIFF函数及用法最佳实践》本文解析DATEDIFF在不同数据库中的差异,强调其边界计算原理,探讨应用场景及陷阱,推荐根据需求选择TIMESTAMPDIFF或inte... 目录1. 核心概念:DATEDIFF 究竟在计算什么?2. 主流数据库中的 DATEDIFF 实现2.1

golang程序打包成脚本部署到Linux系统方式

《golang程序打包成脚本部署到Linux系统方式》Golang程序通过本地编译(设置GOOS为linux生成无后缀二进制文件),上传至Linux服务器后赋权执行,使用nohup命令实现后台运行,完... 目录本地编译golang程序上传Golang二进制文件到linux服务器总结本地编译Golang程序

Linux下删除乱码文件和目录的实现方式

《Linux下删除乱码文件和目录的实现方式》:本文主要介绍Linux下删除乱码文件和目录的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux下删除乱码文件和目录方法1方法2总结Linux下删除乱码文件和目录方法1使用ls -i命令找到文件或目录

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串