struct sockaddr和struct sockaddr_in区别

struct sockaddr和struct sockaddr_in这两个结构体用来处理网络通信的地址。


一、sockaddr


sockaddr在头文件#include <sys/socket.h>中定义,sockaddr的缺陷是:sa_data把目标地址和端口信息混在一起了,如下:

struct sockaddr {  
     sa_family_t sin_family;//地址族
    char sa_data[14]; //14字节,包含套接字中的目标地址和端口信息               
   }; 

 

二、sockaddr_in


sockaddr_in在头文件#include<netinet/in.h>或#include <arpa/inet.h>中定义,该结构体解决了sockaddr的缺陷,把port和addr 分开储存在两个变量中,如下:

struct sockaddr_in
{
    /*地址族,一般来说AF_INET(地址族)PF_INET(协议族)*/
   short sin_family;
    /*端口号,必须要采用网络数据格式,普通数字可以用htons()函数转换成网络数据格式的数字)*/
   unsigned short sin_port;
    /*IP地址, in network byte order(Internet address)*/
   struct in_addr sin_addr;
    /*Same size as struct sockaddr没有实际意义,只是为了跟SOCKADDR结构在内存中对齐*/
   unsigned char sin_zero[8];
};

sin_port和sin_addr都必须是网络字节序(NBO),一般可视化的数字都是主机字节序(HBO)。

其中,结构体in_addr 用来表示一个32位的IPv4地址.
in_addr_t 一般为 32位的unsigned int,其字节顺序为网络顺序(network byte ordered),即该无符号整数采用大端字节序 。.其中每8位代表一个IP地址位中的一个数值.
例如192.168.3.144记为0x9003a8c0,其中 c0 为192 ,a8 为 168, 03 为 3 , 90 为 144
打印的时候可以调用inet_ntoa()函数将其转换为char *类型.

struct in_addr {
    in_addr_t s_addr;
};

使用例子

   struct sockaddr_in servaddr, cliaddr;
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);//将一个32位数从主机字节顺序转换成网络字节顺序
    servaddr.sin_port = htons(PORT); //#define PORT 8889

 

三、总结


二者长度一样,都是16个字节,即占用的内存大小是一致的,因此可以互相转化。二者是并列结构,指向sockaddr_in结构的指针也可以指向sockaddr。

sockaddr常用于bind、connect、recvfrom、sendto等函数的参数,指明地址信息,是一种通用的套接字地址。
sockaddr_in 是internet环境下套接字的地址形式。所以在网络编程中我们会对sockaddr_in结构体进行操作,使用sockaddr_in来建立所需的信息,最后使用类型转化就可以了。一般先把sockaddr_in变量赋值后,强制类型转换后传入用sockaddr做参数的函数:sockaddr_in用于socket定义和赋值;sockaddr用于函数参数。

例子1如下:

#include <stdio.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <netinet/in.h>

int main(int argc,char **argv)
{
    int sockfd;
    struct sockaddr_in mysock;

    sockfd = socket(AF_INET,SOCK_STREAM,0);  //获得fd

    bzero(&mysock,sizeof(mysock));  //初始化结构体
    mysock.sin_family = AF_INET;  //设置地址家族
    mysock.sin_port = htons(800);  //设置端口
    mysock.sin_addr.s_addr = inet_addr("192.168.1.0");  //设置地址
    bind(sockfd,(struct sockaddr *)&mysock,sizeof(struct sockaddr); /* bind的时候进行转化 */
    ... ...
    return 0;
}

例子2如下:

		int socket_fd ;
        unsigned short localPort = 8090;
		unsigned int   localPort =“192.168.0.0”;
		socket_fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
		struct sockaddr_in local_addr;
		local_addr.sin_family = AF_INET;
		local_addr.sin_port = htons(localPort);
		local_addr.sin_addr.s_addr = htonl(localIP);
		rtnValue = bind(socket_fd, reinterpret_cast<struct sockaddr*>(&local_addr), sizeof(struct sockaddr));

		if(0 == rtnValue)
		{
			printf("OpenPort LocalIP = %s LocalPort=%d socket=%d.\r\n", inet_ntoa(local_addr.sin_addr), localPort,socket_fd);
		}
		else
		{
			printf("OpenPort failed!! LocalIP = %s LocalPort=%d socket=%d.\r\n", inet_ntoa(local_addr.sin_addr),localPort,socket_fd);
		}

两个函数 htons() 和 inet_addr()。

端口号转换函数
htons()作用是将端口号由主机字节序转换为网络字节序的整数值。(host to net)

IP地址转换函数
inet_addr()作用是将一个IP字符串转化为一个网络字节序的整数值,用于sockaddr_in.sin_addr.s_addr。
inet_ntoa()作用是将一个sin_addr结构体输出成IP字符串(network to ascii)。比如:
printf("%s",inet_ntoa(mysock.sin_addr));
htonl()作用和htons()一样,不过它针对的是32位的(long),而htons()针对的是两个字节,16位的(short)。

与htonl()和htons()作用相反的两个函数是:ntohl()和ntohs()。


参考:
《TCP/IP网络编程》

  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值