网络编程函数

 
一、网络字节序   
       网络字节顺序是TCP/IP中规定好的一种数据表示格式,它与具体的CPU类型、操作系统等无关,从而可以保证数据在不同主机之间传输时能够被正确解释。网络字节顺序采用big endian排序方式。
       主机字节序就是我们平常说的大端和小端模式:不同的 CPU 有不同的字节序类型,这些字节序是指整数在内存中保存的顺序 这个叫做主机序。
    由于每一台机器内部对变量的字节存储顺序不同,而网络传输的数据是一定要统一顺序的。所以对内部字节表示顺序与网络字节顺序不同的机器,一定要对数据进行转换,从程序的可移植性要求来讲,就算本机的内部字节表示顺序与网络字节顺序相同,也应该在传输数据以前先调用数据转换函数,以便程序移植到其它机器上后能正确执行。真正转换还是不转换是由系统函数自己来决定的。所以在传输多字节整形数时,需要将其转换为网络字节序。 有关的转换函数 
uint16_t htons(uint16_t hostshort);
主机字节顺序转换成网络字节顺序,对无符号短型进行操作
uint32_t htonl(uint32_t hostlong);
主机字节顺序转换成网络字节顺序,对无符号长型进行操作

以上函数的参数就是本地主机序的整数。返回值是:TCP/IP网络字节序

uint16_t ntohs(uint16_t netshort);
网络字节顺序转换成主机字节顺序,对无符号短型进行操作
uint32_t ntohl(uint32_t netlong);
网络字节顺序转换成主机字节顺序,对无符号长型进行操作
注:以上函数原型定义在netinet/in.h里

int inet_aton(const char *cp, struct in_addr *inp);
        功能:将一个字符串IP地址转换为一个32位的网络序列IP地址。
        参数:
                 cp:点分十进制的字符串IP地址
                 inp:存放转换好的32位网络字节序
        返回值:成功返回非0值,如果地址不正确会返回0.并不设置errno

char *inet_ntoa(struct in_addr in);
        功能:将32位的网络字节序转换为字符串IP地址
        参数:
                 in:32位网络字节序IP地址。
        返回值:成功返回指向字符串的指针。错误,返回NULL。
  注:inet_aton和inet_ntoa仅适用于IPV4协议
 
int inet_pton(int af, const char *src, void *dst);
  功能:将一个字符串IP地址转换为一个网络序列IP地址
参数:
              af:AF_INET :需要转换的得IPV4地址,函数将该地址转换为in_addr的结构体
                     AF_INET6:需要转换的是IPV6地址,函数将该地址转换为in6_addr的结构体
              src:字符串IP地址
              dst:存放转换好的IP地址
        返回值:成功返回1,当src为无效值时,返回-1,当af为无效值时,返回0;

const char *inet_ntop(int af, const void *src,char *dst, socklen_t size);
      功能:将网络字节序地址转换为字符串IP地址
  参数:
           af:AF_INET :需要转换的得IPV4地址,函数将该地址转换为in_addr的结构体
                      AF_INET6:需要转换的是IPV6地址,函数将该地址转换为in6_addr的结构体
                src:网络字节序IP地址
                dst:存放转换好的字符串IP地址
size:是所指向缓存区dst的大小,避免溢出,如果缓存区太小无法存储地址的值,则返回一个空指针
 返回值:成功返回一个指向dst的非空指针,失败返回一个NULL

二、其他函数:
网络编程的一些结构体:
     struct sockaddr {
             unsigned short sa_family;      是2字节的地址家族,一般都是“AF_xxx”的形式。
       char            sa_data[14];   sa_data : 是14字节的协议地址。   };
由于上面的通用结构体易用性不是很好,所以一般使用下面的结构体。
struct sockaddr_in {
  short int sin_family;      指代协议族
  unsigned short int sin_port;      存储端口号(使用网络字节顺序)
  struct in_addr sin_addr;        存储IP地址,使用in_addr这个数据结构
  unsigned char sin_zero[8];     是为了让sockaddr与sockaddr_in两个数据结构保持大小相同而保留的空字节。
  };
typedef struct in_addr {
 unsigned long s_addr;       用一个长整型来表示IP地址。
 
int socket(int domain, int type, int protocol);
        功能:创建一个套接字
        参数:
                domain:AF_INET:   IPV4协议
                                 AF_INET6:IPV6协议
                                 AF_LOCAL: Unix域协议
                                 AF_ROUTE:路由套接口
                                AF_KEY:密钥套接口
                type:SOCK_STREAM:字节流套接口  ,面向连接的,可靠地字节流。TCP协议
                        SOCK_DGRAM:数据报套接口,无连接的不可靠报文传递。UDP协议
                        SOCK_RAW:原始套接口,IP协议的数据报接口,需要超级用户权限。绕过TCP,UDP传输协议。
            protocol:一般为0,当使用原始套接字时除外。
       返回值:成功返回一个新的套接字描述符,失败返回-1.

int bind(int sockfd, const struct sockaddr *addr,socklen_t addrlen);
        功能将一本地地址与一套接口捆绑
        参数:
                 sockfd 表示已经建立的socket编号(描述符)
 addr: 是一个指向sockaddr结构体类型的指针;如果只为INADDR_ANY,套接字可以被绑定到系统的任一可用的网络地址,这意味着可以收到这个系统所安装的所有网卡的数据包。
     
     
     
     
                     addrlen:表示struct sockaddr结构的长度,可以用sizeof函数获得。
        返回值:成功返回0,失败返回-1;
注:端口号一般不要小于5000,当端口设置为0时, 随机选择一个没有使用的端口。当使用INADDR_ANY时,这时bind其实并没有绑定。 

int getsockname(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
        功能:用于获取一个已绑定到套接字的地址(IP地址和端口号等)
      参数:
                   sockfd:套接字描述符
                   addr:将返回的地址放到addr中。
addrlen:该数在调用函数以前被设置为struct sockaddr的长度,当函数返回时,该数会被设为struct sockaddr的大小,如果两者不匹配,则将其截短不报错。
        返回值:成功返回0,出错返回-1
注:如果当前套接字没有绑定任何地址,则其结果没有定义。 若一个套接口与INADDR_ANY捆绑,也就是说该套接口可以用任意主机的地址,此时除非调用connect()或accept()来连接,否则getsockname()将不会返回主机IP地址的任何信息

int   getpeername(int   sockfd,  struct  sockaddr  *addr,socklen_t *addrlen);
        功能:若套接字已经和对方连接,获得对方的地址(IP地址和端口号等)
        参数:
      
      
      
      
                   sockfd:套接字描述符
                   addr:将返回的地址放到addr中。
     
     
     
     
addrlen:该数在调用函数以前被设置为struct sockaddr的长度,当函数返回时,该数会被设为struct sockaddr的大小,如果两者不匹配,则将其截短不报错。

      返回值:成功返回0,出错返回-1
     注:该套接字必须已与一个地址连接,否则返回不确定。

int connect(int sockfd, const struct sockaddr *addr,socklen_t addrlen);
        功能:在使用面向连接的网络服务以前,需在客户端和服务器端建立一个连接,可以使用connect来建立连接
        参数:
                sockfd: 套接字文件描述符
                addr: 要连接的目的地址
                addrlen:struct sockaddr的长度
     返回值:成功返回0,失败返回-1
注:
     
     
     
     
  • 如果sockfd并没有绑定到一个地址,connect会给调用者绑定一个默认地址。如果connect失败,则套接口不可再用,必须关闭,不能在对此套接口调用connect。必须关闭后,重新调用socket创建一个套接口
  •  connect还可以用于无连接的网络服务,如果在SOCK_DGRAM套接字上调用connect,所有发送报文的目标地址设为connect调用中所指定的地址,这样,每次在传送报文时,就不需要再提供地址。另外,仅能接受来自指定地址的报文。

int listen(int sockfd, int backlog);
        功能:设置可以监听最大连接请求,包括已连接的和未连接的
        参数:
                    sockfd:套接字描述符
                    backlog:最大连接请求数
        返回值:成功返回0,失败返回-1
注:当socket创建一个套接口时,它被假设为一个主动接口,也就是说,他是一个将调用connect发起连接的客户套接口,函数listen将未连接的套接口转换成被动套接口,只是内核应接受向此套接口的连接请求。调用listen导致套接口从CLOSED状态转换到LISTEN状态。一旦调用了listen,套接字就能接受连接请求。
不要将backlog定义为0,一位不同实现对此有不同解释,有些实现容许有一个连接排队,有的则不容许有连接排队。

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
        功能:等待接受连接请求。
        参数:
                sockfd:套接字描述符
                 addr:用来存放返回的客户端地址
addrlen:在此函数调用前,将addrlen设置为addr所指向的套接字地址结构的长度。当此函数返回后,此数便被设置成由内核定义的此套接字地址结构的准确长度。
返回值:成功返回一个新的套接字描述符,该描述符连接到调用connect的客户端,这个新的描述符合原始套接字(sockfd)具有相同的套接字类型和地址族。以后对连接的客户端操作时,便使用这个新的描述符。失败返回-1.
     注:如果当前没有未完成的连接请求,则此函数睡眠阻塞。如果sockfd处于非阻塞模式下,则会返回-1。因为一般情况下服务器创建的sockfd 还要用于接收其它客户端的连接请求。所以accept要返回一个新的描述符。而sockfd继续等待别的连接请求。而connect用于客户端,一般只与一个服务器连接,所以不返回一个新的描述符

三、数据传送函数
ssize_t send(int sockfd, const void *buf, size_t len, int flags);
            功能:发送数据
            参数:
                  sockfd:套接字描述符
                      buf:待发送数据的缓冲区
                      len:要发送的数据长度
                     flags:
                        MSG_OOB 发送带外数据。
                        MSG_DONTROUTE 勿将数据路由出本地网络
MSG_DONTWAIT 允许非阻塞操作
        返回值:成功则返回实际传送出去的字符数,失败返回-1
        注:使用send时,套接字必须是已连接的,因为send不知道要将数据发送何处。当套接字输出队列没有足够的空间来发送消息时,send会阻塞,同样适用于下列两个函数sendto和sendmsg。但当套接字在非阻塞模式下士时,情况会改变,在这种情况下,这些函数不会阻塞而是失败。

 
ssize_t sendto(int  sockfd, const void *buf, size_t len, int  flags,const struct sockaddr *dest_addr, socklen_t  addrlen);
            功能:发送数据
            参数:
                     sockfd:套接字描述符
       
       
       
       
           buf:待发送数据的缓冲区
           len:要发送的数据长度
           flags:
     
     
     
     
   MSG_OOB 发送带外数据。
     
     
     
     
   MSG_DONTROUTE 勿将数据路由出本地网络
     
     
     
     
   MSG_DONTWAIT 允许非阻塞操作
          dest_addr:将要发送的目标地址
          addrlen:struct sockaddr的长度
      返回值:成功则返回实际传送出去的字符数,失败返回-1

ssize_t sendmsg(int sockfd, const struct msghdr *msg, int flags);
            功能:发送数据
          参数:
        sockfd:套接字描述符
        msg:
             struct msghdr {
                                            void         *msg_name;
                                            socklen_t    msg_namelen;
                                           struct iovec *msg_iov;
                                            size_t       msg_iovlen;
                                            void         *msg_control;
                                            size_t       msg_controllen;
                                            int          msg_flags;
                                };
        flags:
MSG_OOB 发送带外数据。
MSG_DONTROUTE 勿将数据路由出本地网络
MSG_DONTWAIT 允许非阻塞操作
        返回值:成功则返回实际传送出去的字符数,失败返回-1
     注:
       
       
       
       
  • 对于以上三个函数,当套接字输出队列没有足够的空间来发送消息时,则会一直阻塞。但当套接字在非阻塞模式下时,情况会改变,在这种情况下,这些函数不会阻塞而是失败。
  • 如果调用shutdow关闭了读端或者服务器端进程关闭socket以后,则当前进程收到一个SIGPIPE信号,终止进程
ssize_t recv(int sockfd, void *buf, size_t len, int flags);
      
      
      
      
      功能:接收数据
      参数:
                   sockfd: 套接字描述符
                   buf:         接收数据的缓冲区
                   len:          要接收的数据长度
                   flags:
                MSG_OOB 接收带外数据。
      
      
      
      
MSG_DONTWAIT 允许非阻塞操作
      
      
      
      
MSG_TRUNC  一般数据被截断
      
      
      
      
MSG_TRUNC  控制数据被截断
      
      
      
      
MSG_TRUNC  控制数据被截断
      
      
      
      
MSG_EOR  接收到记录结束符
             返回值:成功返回读到的字节数,失败返回-1.
注:如果调用shutdow关闭了发送端,则当所有数据接收完毕后,返回0;当客户端进程关闭时,ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,struct sockaddr *src_addr, socklen_t *addrlen);
     
     
     
     
     功能:接收数据
     参数:
                  sockfd:   套接字描述符
                   buf:    接收数据的缓冲区
                  len:      要接收的数据长度
flags:
      
      
      
      
    MSG_OOB 接收带外数据。
MSG_DONTWAIT 允许非阻塞操作
MSG_TRUNC 一般数据被截断
MSG_TRUNC 控制数据被截断
MSG_EOR 接收到记录结束符
     
     
     
     
src_addr:将返回的发送者地址存放在src_addr中。
addrlen:在此函数调用前,将addrlen设置为addr所指向的套接字地址结构的长度。当此函数返回后,此数便被设置成由内核定义的此套接字地址结构的准确长度。

          返回值:成功返回读到的字节数,失败返回-1.

ssize_t recvmsg(int sockfd, struct msghdr *msg, int flags);
        功能:接收数据
     
     
     
     
         
         
         
         
    参数:
           sockfd:套接字描述符
                 msg:
                          struct msghdr {
                                            void         *msg_name;
                                            socklen_t    msg_namelen;
                                           struct iovec *msg_iov;
                                            size_t       msg_iovlen;
                                            void         *msg_control;
                                            size_t       msg_controllen;
                                            int          msg_flags;
                                };
             flags:
                        MSG_OOB 接收带外数据。
                    MSG_DONTWAIT 允许非阻塞操作
                    MSG_TRUNC  一般数据被截断
                    MSG_TRUNC  控制数据被截断
                    MSG_EOR  接收到记录结束符
        返回值:成功返回读到的字节数,失败返回-1.
注:
     
     
     
     
  • 当服务器的sockfd和地址绑定以后,以上三个recv函数,可以接受任何向此地址发送数据的连接,并没有指定要接受的地址。
  • 以上三个函数当为阻塞模式下时,如果没有数据可读,则阻塞。如果调用shutdow关闭了发送端或者客户端进程关闭socket以后,则当所有数据接收完毕后,返回0;
   
   
   
   
 
四、套接字选项:
int getsockopt(int sockfd, int level, int optname,void *optval, socklen_t *optlen);
        功能:获得套接字的选项(属性)
        参数:
                    socket:套接字描述符   
                    level:选项所在的协议层。见下表1.1
                    optname:选项名称,见下表1.1
                   optval:将获得的相应选项的开关状态存放在optval中,如果为0,表示此选项关闭,非0,表示此选项被启用。
                                optval必须指向每个选项的数据类型。
                    optlen:optval 的大小
     返回值:成功返回0,出错返回-1

 int setsockopt(int sockfd, int level, int optname,const void *optval, socklen_t optlen);             功能:设置一个套接字的选项(属性)
          参数:
    socket:套接字描述符
          level:选项所在的协议层。见下表1.1
         optname:选项名称,见下表1.1
      optval:没有特殊说明时,默认为整形设置相应选项的开关状态,如果为0,表示要将此选项关闭,非0,表示要将此选项启用。如果为特殊说明结构,则需要按照结构设置。 
            optval必须指向每个选项的数据类型。
         optlen:optval 的大小
返回值:成功返回0,失败返回-1.
       
       
       
       
                                                表1.1
网络编程函数_第1张图片


 
 
 









 
 















你可能感兴趣的:(网络)