UNP——第三章,套接字编程介绍

1.套接字结构

  多数套接字函数都有套接字结构参数,每个协议族都定义了自己的套接字结构,以 sockaddr_ 开始,并对应协议族的唯一后缀。

  如 IPv4 sockaddr_in

    IPv6 sockaddr_in6

      Unix sockaddr_un

           链路   sockaddr_dl

           存储   sockaddr_storage

struct sockaddr_in {
    uint8_t              sin_len;
    sa_family_t          sin_family;   
    in_port_t            sin_port;
    struct in_addr       sin_addr;
    char                 sin_zero[8];  
}

  其中 sin_len 用于内核层,在应用读写都无意义。

  sin_zero 没有用处。

  POSIX也要求要定义 sin_family, sin_addr, sin_port

 

2.传参

  套接字调用,参数以引用方式,在内核和应用传递。

  当参数 ,应用 ----> 内核,需要传 len_of_socket

    如: bind, connect, sendto

                 内核 -----> 应用,需要传 &len_of_socket

    即 len_of_socket 做输入,输出参数。

 

3.字节序

  网络字节序为大端,即先存储数据高位。

  小端,则为 先存储数据地位。

  可以如下判断主机字节序:

union {
    short a;
    char  b[sizeof(short)];
}c = {0x01, 0x00};

if (c.b[0] == 1 && c.b[1] == 0) {
    printf("小端\n");
}
else if (c.b[0] == 0 && c.b[1] == 1) {
    printf("大端\n");
}
else {
    printf("未知\n")
}

  对于大小端转换函数,推荐 inet_pton, inet_ntop

  不推荐 inet_aton, inet_ntoa,因为不支持 IPv6

 

3.封装api

  为了兼容多种情况,可以进行如下封装:

char *
sock_ntop(const struct sockaddr *sa, socklen_t salen)
{
       switch(sa->sa_family) {
          case AF_INET:
              .....
           beak;
          ....    
       }
}

  另外,对套接字的读写不同于一般文件读写,套接字读写,即使返回值未达到预期,也不一定错误,因为可能是缓冲区满了,就需要将剩下的数据继续操作。

  类似的,其他缓冲区(如管道)读写也是这样。

  所以需要封装 readn, writen 等函数。

你可能感兴趣的:(UNP——第三章,套接字编程介绍)