TCP/UDP中sockaddr和sockaddr_in的区别及用法

TCP/UDP中sockaddr和sockaddr_in的区别及用法目录一、先看两个结构体的定义1.1sockaddr1.2sockaddr_in二、sockaddr与sockaddr_in的相同点和区别2.1相同点2.2区别三、使用方法3.1socket编程中sockaddr和sockaddr_in的强制转换3.2补充部分structsockaddr和structsockaddr_in这两个结构体用来处理网络通信的地址。在各种系统调用或者函数中,只要和网络地址打交道,就得用到这两个结构体。网络中的地址包含3个方面.

大家好,欢迎来到IT知识分享网。TCP/UDP中sockaddr和sockaddr_in的区别及用法"

目录

一、先看两个结构体的定义

1.1 sockaddr

1.2 sockaddr_in

二、sockaddr与sockaddr_in的相同点和区别

2.1 相同点

2.2 区别

三、使用方法

3.1 socket编程中sockaddr和sockaddr_in的强制转换

3.2 补充部分


struct sockaddr和struct sockaddr_in这两个结构体用来处理网络通信的地址。
在各种系统调用或者函数中,只要和网络地址打交道,就得用到这两个结构体。
网络中的地址包含3个方面的属性:
1.地址类型: ipv4还是ipv6
2.ip地址
3.端口

一、先看两个结构体的定义

1.1 sockaddr

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

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

1.2 sockaddr_in

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

struct sockaddr_in {
    short            sin_family;       // 2 bytes e.g. AF_INET, AF_INET6
    unsigned short   sin_port;    //16位 2 bytes e.g. htons(3490)
    struct in_addr   sin_addr;     //32位 4 bytes see struct in_addr, below
    char             sin_zero[8];     // 8 bytes zero this if you want to
};
//另一个结构体 in_addr存放32位ip地址
struct in_addr {
    unsigned long s_addr;          // 4 bytes load with inet_pton()
};

二、sockaddr与sockaddr_in的相同点和区别

2.1 相同点

两个结构体的大小都是16个字节,而且都有family属性;

2.2 区别

数据形式上的区别:

sockaddr用14个字节来表示sa_data,而sockaddr_in把14个字节拆分成sin_port, sin_addr和sin_zero分别表示端口、ip地址。sin_zero用来填充字节使sockaddr_in和sockaddr保持一样大小。
sockaddr和sockaddr_in包含的数据都是一样的。

使用上的区别:

程序员不应操作sockaddr,sockaddr是给操作系统用的
程序员应使用sockaddr_in来表示地址,sockaddr_in区分了地址和端口,使用更方便。

也就是,程序员把类型、ip地址、端口填充sockaddr_in结构体然后强制转换成sockaddr,作为参数传递给系统调用函数。

三、使用方法

3.1 socket编程中sockaddr和sockaddr_in的强制转换

先看看网络编程经典用法。

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

#define SERV_IP "192.168.1.0"   //服务器ip

int main(int argc,char **argv)
{
    int sockfd;
    struct sockaddr_in mysock;//定义sockaddr_in结构体mysock

    sockfd = socket(AF_INET,SOCK_STREAM,0);  //定义套接字 获得sockfd

    /******填充sockaddr_in******/
    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");  //设置ip地址
    //mysock.sin_addr.s_addr = inet_addr(SERV_IP); //将字符串形式的ip地址转换为点分十进制格式的ip地址


    /******绑定端口的时候进行强制转换******/
    bind(sockfd,(struct sockaddr *)&mysock,sizeof(struct sockaddr); //bind的时候进行转换 

    ... ...

    /******连接的时候进行强制转换********/
    connect(sockfd, (struct sockaddr *) &mysock, sizeof(mysock));

    ... ...
    return 0;
}

3.2 补充部分

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

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

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()。

免责声明:本站所有文章内容,图片,视频等均是来源于用户投稿和互联网及文摘转载整编而成,不代表本站观点,不承担相关法律责任。其著作权各归其原作者或其出版社所有。如发现本站有涉嫌抄袭侵权/违法违规的内容,侵犯到您的权益,请在线联系站长,一经查实,本站将立刻删除。 本文来自网络,若有侵权,请联系删除,如若转载,请注明出处:https://yundeesoft.com/21520.html

(0)

相关推荐

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注

关注微信