2.套接口编程相关的辅助函数

    技术2022-05-11  64

    2.1 字节排序问题

    考虑一个16位整数,他由2个字节组成。内存中存储这两个字节有两中方法:

    一.将低序字节存储在起始地址,这称为小端字节序(网络字节序)。 二.将高序字节存储在起始地址,这称为大端字节序(主机字节序)。   2.2 字节排序函数 h 代表 host, n 代表 network L 代表 long 32 Ipv4 地址), s 代表 short 16 TCP 或者 UDP 端口号)        #include <netinet/in.h>       uint16_t htons(uint16_t host16bitvalue);    // 将主机字节序的端口号转换成套接口地址中的 网络字节序端口        uint32_t htonl(uint32_t host32bitvalue);     // 同上,转换 Ipv4 地址        uint16_t ntohs(uint16_t net16bitvalue);       uint32_t ntohl(uint32_t net32bitvalue); 2.3 字节操纵函数 #include <strings.h> void bzero(void *dest, size_t nbytes); void bcopy(const void *src, void *dest, size_t nbytes); void bcmp(const void *ptr1, const void *ptr2, size_t nbytes);  返回 0 相等,非 0 不相等 bzero 将目标中指定数目的字节置为 0 ,常常用来将套接口地址结构初始化为 0 bcopy 将指定数目的字节从源移到目标。 bcmp 比较任意两个字节串   #include <string.h> void *memset(void *dest, int c, size_t len); void *memcpy(void *dest, const void *src, size_t nbytes); void memcmp(const void *ptr1, const void *ptr2, size_t nbytes); 返回 0 相同,非 0 不相同 memset 将目标中指定数目的字节置 c   一般选者 bzero ,因为参数比较少。 memcpy 注意该函数的 dest src 位置与 bcopy 不同。   2.4 地址转换函数 该系列函数负责在 ASCII 字符串(人们比较喜欢用的格式: 192.168.0.45 )与网络字节序的二进制值(此值存于套接口地址结构中)间转换地址。 p 代表 ASCII 字符串, n 代表套接口地址结构中的二进制值 #include <arpa/inet.h> int inet_pton(int family, const char *strptr, void *addrptr); 返回 1 成功, 0 无效的表达式, -1 出错 const char *inet_ntop(int family, const void *addrptr, char *strptr, size_t len);                                                         返回指向结果的指针成功, NULL 出错 Family可以是AF_INET,也可以是 AF_INET6。 函数inet_ntop的参数strptr不能是个空指针,调用者必须为目标分配内存并指定大小。成功时此指针即函数的返回值。   2.5 字节流套接口上的read和write函数 : readn、writen 字节流套接口上的读或写输入或输出的字节数可能比要求的数量少,但这不是错误状况,原因是内核中套接口的缓冲区可能已达到极限。 此时需要调用者再次调用read和write函数,以输入和输出剩余的字节。 这种情况在读字节流套接口时很常见,但在写字节流套接口时只能在套接口非阻塞的情况下才出现。 ssize_t readn(int filedes, void *buff, size_t nbytes); ssize_t writen(int filedes, const void *buff, size_t nbytes);   ssize_t readn(int fd, void *vptr, size_t n) {                                  //不一定能读到n个        size_t   nleft;        ssize_t nread;        char    *ptr;               ptr = vptr;        nleft = n;        while ( nleft > 0 )        {              if ( ( nread = read ( fd, ptr, nleft ) ) < 0 )               {                      if ( errno == EINTR )                      {                             nread = 0;     //中断重启(重读)                      }                      else                      {                             return ( -1 );                      }            else               {                      if ( nread == 0)                      {                             break;       //EOF 字节流结束了                      }               }                             nleft -= nread;               ptr += nread;         }  return ( n – nleft );         //返回读到的实际字节数目 }     ssize_t writen(int fd, const void *vptr, size_t n) {        size_t    nleft;        ssize_t    nwritten;        const char *ptr;          ptr = vptr;        nleft = n;        while ( nleft > 0 )        {               if ( ( nwritten = write ( fd, ptr, nleft) ) <= 0 )               {                      if ( nwritten < 0 && errno == EINTR )                      {                             nwritten = 0;         //中断重启                      }                      else                      {                             return ( -1 );          //error                       }                }                             nleft -= nwritten;               ptr += nwritten;      }               return ( n ); }

    最新回复(0)