1.socket编程:socket编程,网络字节序,函数介绍,IP地址转换函数,sockaddr数据结构,网络套接字函数

更新时间:2023-11-24 12:43:01 阅读量: 教育文库 文档下载

说明:文章内容仅供预览,部分内容可能不全。下载后的文档,内容与下面显示的完全一致。下载之前请确认下面内容是否您想要的,是否完整无缺。

1Socket编程

socket这个词可以表示很多概念: 在TCP/IP协议中,“IP地址+TCP或UDP端口号”唯一标识网络通讯中的一个进程,“IP 地址+端口号”就称为socket。

在TCP协议中,建立连接的两个进程各自有一个socket来标识,那么这两个socket组成的socket pair就唯一标识一个连接。socket本身有“插座”的意思,因此用来描述网络连 接的一对一关系。

TCP/IP协议最早在BSD UNIX上实现,为TCP/IP协议设计的应用层编程接口称为socket API。

本章的主要内容是socket API,主要介绍TCP协议的函数接口,最后介绍UDP协议和UNIXDomain Socket的函数接口。

图11.1:socketAPI

2 网络字节序

我们已经知道,内存中的多字节数据相对于内存地址有大端和小端之分,磁盘文件中的 多字节数据相对于文件中的偏移地址也有大端小端之分。网络数据流同样有大端小端之分, 那么如何定义网络数据流的地址呢?发送主机通常将发送缓冲区中的数据按内存地址从低到高的顺序发出,接收主机把从网络上接到的字节依次保存在接收缓冲区中,也是按内存地址从低到高的顺序保存,因此,网络数据流的地址应这样规定:先发出的数据是低地址,后发出的数据是高地址。

TCP/IP协议规定,网络数据流应采用大端字节序,即低地址高字节。例如上一节的UDP

段格式,地址0-1是16位的源端口号,如果这个端口号是1000(0x3e8),则地址0是0x03, 地址1是0xe8,也就是先发0x03,再发0xe8,这16位在发送主机的缓冲区中也应该是低地址存0x03,高地址存0xe8。但是,如果发送主机是小端字节序的,这16位被解释成0xe803,而不是1000。因此,发送主机把1000填到发送缓冲区之前需要做字节序的转换。同样地,接收主机如果是小端字节序的,接到16位的源端口号也要做字节序的转换。如果主机是大端字节序的,发送和接收都不需要做转换。同理,32位的IP地址也要考虑网络字节序和主机字节序的问题。

为使网络程序具有可移植性,使同样的C代码在大端和小端计算机上编译后都能正常运行,可以调用以下库函数做网络字节序和主机字节序的转换。 3函数介绍

A 依赖的头文件

#include

B 函数声明

#include

uint32_thtonl(uint32_t hostlong); uint16_thtons(uint16_t hostshort); uint32_tntohl(uint32_t netlong); uint16_tntohs(uint16_t netshort);

h表示host,n表示network,l表示32位长整数,s表示16位短整数。 如果主机是小端字节序,这些函数将参数做相应的大小端转换然后返回,如果主机是大端字节序,这些函数不做转换,将参数原封不动地返回。

uint32_thtonl(uint32_t hostlong); 名称: htonl 功能: The htonl() function converts the unsigned integer hostlong from hostbyte order to network byte order 头文件: #include 函数原形: uint32_t htonl(uint32_t hostlong); 参数: 返回值: uint16_thtons(uint16_t hostshort); 名称: htons 功能: The htons() function converts the unsigned short integer hostshort from host byte order to network byte order. 头文件: #include 函数原形: uint16_t htons(uint16_t hostshort); 参数: 返回值:

uint32_tntohl(uint32_t netlong); 名称: ntohl 功能: The ntohl() function converts the unsigned integer netlong from network byte order to host byte order. 头文件: #include 函数原形: uint32_t ntohl(uint32_t netlong); 参数: 返回值:

uint16_tntohs(uint16_t netshort); 名称: ntohs 功能: The ntohs() function converts the unsigned short integer netshort from network byte order to host byte order. 头文件: #include 函数原形: uint16_t ntohs(uint16_t netshort); 参数: 返回值: 4 IP地址转换函数 #include #include #include

intinet_aton(const char *cp, structin_addr *inp); in_addr_tinet_addr(const char *cp); char *inet_ntoa(structin_addr in); 只能处理IPv4的ip地址 不可重入函数

注意参数是structin_addr 现在

#include

intinet_pton(intaf, const char *src, void *dst);

const char *inet_ntop(intaf, const void *src, char *dst, socklen_t size);

支持IPv4和IPv6 可重入函数

其中inet_pton和inet_ntop不仅可以转换IPv4的in_addr,还可以转换IPv6的in6_addr,因此函数接口是void *addrptr

5 sockaddr数据结构

strcutsockaddr很多网络编程函数诞生早于IPv4协议,那时候都使用的是sockaddr结

构体,为了向前兼容,现在sockaddr退化成了(void *)的作用,传递一个地址给函数,至 于这个函数是sockaddr_in还是sockaddr_in6,由地址族确定,然后函数内部再强制类型转 化为所需的地址类型

图 11.2:sockaddr数据结构

structsockaddr {

sa_family_tsa_family; /* address family, AF_xxx */ charsa_data[14]; /* 14 bytes of protocol address */ };

structsockaddr_in {

__kernel_sa_family_tsin_family; /* Address family */ __be16 sin_port; /* Port number */

structin_addrsin_addr; /* Internet address */ /* Pad to size of `structsockaddr'. */

unsigned char __pad[__SOCK_SIZE__ - sizeof(short int) - sizeof(unsigned short int) - sizeof(structin_addr)]; };

/* Internet address. */ structin_addr {

__be32 s_addr; };

struct sockaddr_in6 {

unsigned short int sin6_family; /* AF_INET6 */

__be16 sin6_port; /* Transport layer port # */ __be32 sin6_flowinfo; /* IPv6 flow information */ struct in6_addr sin6_addr; /* IPv6 address */

__u32 sin6_scope_id; /* scope id (new in RFC2553) */ };

struct in6_addr { union {

__u8 u6_addr8[16]; __be16 u6_addr16[8]; __be32 u6_addr32[4]; } in6_u;

#define s6_addr in6_u.u6_addr8 #define s6_addr16 in6_u.u6_addr16 #define s6_addr32 in6_u.u6_addr32 };

#define UNIX_PATH_MAX 108 structsockaddr_un {

__kernel_sa_family_tsun_family; /* AF_UNIX */ charsun_path[UNIX_PATH_MAX]; /* pathname */ };

Pv4和IPv6的地址格式定义在netinet/in.h中,IPv4地址用sockaddr_in结构体表示,包 括16位端口号和32位IP地址,IPv6地址用sockaddr_in6结构体表示,包括16位端口号、128位IP地址和一些控制字段。UNIX Domain Socket的地址格式定义在sys/un.h中,用sockaddr_un结构体表示。各种socket地址结构体的开头都是相同的,前16位表示整个结构体的长度(并不是所有UNIX的实现都有长度字段,如Linux就没有),后16位表示地址类型。IPv4、IPv6和Unix Domain Socket的地址类型分别定义为常数AF_INET、AF_INET6、AF_UNIX。这样,只要取得某种sockaddr结构体的首地址,不需要知道具体是哪种类型的sockaddr结构体,就可以根据地址类型字段确定结构体中的内容。因此,socket API可以接受各种类型的sockaddr结构体指针做参数,例如bind、accept、connect等函数,这些函数的参数应该设计成void *类型以便接受各种类型的指针,但是sock API的实现早于ANSI C标准化,那时还没有void *类型,因此这些函数的参数都用structsockaddr *类型表示,在传递参数之前要强制类型转换一下,例如: structsockaddr_inservaddr; /* initialize servaddr*/

bind(listen_fd,(structsockaddr *)&servaddr,sizeof(servaddr));

6 网络套接字函数 A socket

#include #include

intsocket(intdomain,inttypes,int protocol);

domain:

AF_INET 这是大多数用来产生socket的协议,使用TCP或UDP来传输,用IPv4的地址 AF_INET6 与上面类似,不过是来用IPv6的地址

本文来源:https://www.bwwdw.com/article/ghzt.html

Top