linux C网络编程基本流程

Linux C编程---网络编程

         网络编程,一定离不开套接口;那什么是套接口呢?在Linux下,所有的I/O操作都是通过读写文件描述符而产生的,文件描述符是一个和打开的文件相关联的整数,这个文件并不只包括真正存储在磁盘上的文件,还包括一个网络连接、一个命名管道、一个终端等,而套接口就是系统进程和文件描述符通信的一种方法。目前最常用的套接口是字:字节流套接口(基于TCP)和数据报套接口(基于UDP),当然还有原始套接口(原始套接口提供TCP套接口和UDP套接口所不提供的功能,如构造自己的TCP或UDP分组)等,我们这里主要介绍字节流套接口和数据报套接口。

         1、socket函数:为了执行网络输入输出,一个进程必须做的第一件事就是调用socket函数获得一个文件描述符。

         

[cpp] view plain copy print ?
  1. #include <sys/socket.h>   
  2. int socket(int family,int type,int protocol);      
  3.      返回:非负描述字---成功   -1---失败  
        第一个参数指明了协议簇,目前支持5种协议簇,最常用的有AF_INET(IPv4协议)和AF_INET6(IPv6协议);第二个参数指明套接口类型,有三种类型可选:SOCK_STREAM(字节流套接口)、SOCK_DGRAM(数据报套接口)和SOCK_RAW(原始套接口);如果套接口类型不是原始套接口,那么第三个参数就为0。

        2、connect函数:当用socket建立了套接口后,可以调用connect为这个套接字指明远程端的地址;如果是字节流套接口,connect就使用三次握手建立一个连接;如果是数据报套接口,connect仅指明远程端地址,而不向它发送任何数据。

       

[cpp] view plain copy print ?
  1. #include <sys/socket.h>         
  2.   int connect(int sockfd,const struct sockaddr * servaddr,socklen_t  
  3.  addrlen);    
  4.            返回:0---成功   -1---失败  

          第一个参数是socket函数返回的套接口描述字;第二和第三个参数分别是一个指向套接口地址结构的指针和该结构的大小。

          这些地址结构的名字均已“sockaddr_”开头,并以对应每个协议族的唯一后缀结束。以IPv4套接口地址结构为例,它以“sockaddr_in”命名,定义在头文件<netinet/in.h>;以下是结构体的内容:

         

[cpp] view plain copy print ?
  1. struct in_addr {  
  2.  in_addr_t s_addr;      /* IPv4地址 */  
  3. };  
  4. struct sockaddr_in {  
  5.  uint8_t sin_len; /* 无符号的8位整数 */  
  6. sa_family_t sin_family;  
  7. /* 套接口地址结构的地址簇,这里为AF_INET */  
  8. in_port_t sin_port; /* TCP或UDP端口 */  
  9. struct in_addr sin_addr;  
  10.  char sin_zero[8];  
  11.  };     

          3、bind函数:为套接口分配一个本地IP和协议端口,对于网际协议,协议地址是32位IPv4地址或128位IPv6地址与16位的TCP或UDP端口号的组合;如指定端口为0,调用bind时内核将选择一个临时端口,如果指定一个通配IP地址,则要等到建立连接后内核才选择一个本地IP地址。

          

[cpp] view plain copy print ?
  1. #include <sys/socket.h>     
  2.  int bind(int sockfd,const struct sockaddr * myaddr,socklen_t  
  3.  addrlen);  
  4.  返回:0---成功   -1---失败   
            第一个参数是socket函数返回的套接口描述字;第二和第第三个参数分别是一个指向特定于协议的地址结构的指针和该地址结构的长度。

          4、listen函数:listen函数仅被TCP服务器调用,它的作用是将用sock创建的主动套接口转换成被动套接口,并等待来自客户端的连接请求。

         

[cpp] view plain copy print ?
  1. #include <sys/socket.h>   
  2.  int listen(int sockfd,int backlog);     
  3.  返回:0---成功   -1---失败  
  4.  -------------------------------------------------------------------  
           第一个参数是socket函数返回的套接口描述字;第二个参数规定了内核为此套接口排队的最大连接个数。由于listen函数第二个参数的原因,内核要维护两个队列:以完成连接队列和未完成连接队列。未完成队列中存放的是TCP连接的三路握手为完成的连接,accept函数是从以连接队列中取连接返回给进程;当以连接队列为空时,进程将进入睡眠状态。

          5、accept函数:accept函数由TCP服务器调用,从已完成连接队列头返回一个已完成连接,如果完成连接队列为空,则进程进入睡眠状态。

          

[cpp] view plain copy print ?
  1. #include <sys/socket.h>            
  2.  int accept(int sockfd,struct sockaddr *  
  3.  cliaddr,socklen_t * addrlen);    
  4.   回:非负描述字---成功   -1---失败  
         6、inet_pton函数:将点分十进制串转换成网络字节序二进制值,此函数对IPv4地址和IPv6地址都能处理

[cpp] view plain copy print ?
  1. #include <arpa/inet.h>   
  2. int inet_pton(int family,const char * strptr,void * addrptr);  
  3.  返回:1---成功 0---输入不是有效的表达格式 -1---失败  
  4.  -------------------------------------------------------------------  

第一个参数可以是AF_INET或AF_INET6:第二个参数是一个指向点分十进制串的指针:第三个参数是一个指向转换后的网络字节序的二进制值的指针。

         7、inet_ntop函数:和inet_pton函数正好相反,inet_ntop函数是将网络字节序二进制值转换成点分十进制串。

        

[cpp] view plain copy print ?
  1. #include <arpa/inet.h>     
  2.  const char * inet_ntop(int family,const void *  
  3. addrptr,char * strptr,size_t len);    
  4.     返回:指向结果的指针---成功   NULL---失败  

第一个参数可以是AF_INET或AF_INET6:第二个参数是一个指向网络字节序的二进制值的指针;第三个参数是一个指向转换后的点分十进制串的指针;第四个参数是目标的大小,以免函数溢出其调用者的缓冲区。

        8、fock函数:在网络服务器中,一个服务端口可以允许一定数量的客户端同时连接,这时单进程是不可能实现的,而fock就分配一个子进程和客户端会话,当然,这只是fock的一个典型应用。

      

[cpp] view plain copy print ?
  1. #include <unistd.h>     
  2.  pid_t fock(void);  
  3. 返回:在子进程中为0,在父进程中为子进程ID   -1---失败  
  4.  -------------------------------------------------------------------  

下面就用一个例子来说明服务器和客户是怎么连接的

[cpp] view plain copy print ?
  1. * client.c */  
  2. #include <stdio.h>   
  3. #include <stdlib.h>   
  4. #include <errno.h>   
  5. #include <string.h>   
  6. #include <netdb.h>   
  7. #include <sys/types.h>   
  8. #include <netinet/in.h>   
  9. #include <sys/socket.h>   
  10. int main(int argc,char *argv[]) {  
  11. int sockfd,numbytes;  
  12. char buf[100];  
  13. struct hostent *he;  
  14. struct sockaddr_in their_addr;  
  15. int i = 0;  
  16. //将基本名字和地址转换   
  17. he = gethostbyname(argv[1]);  
  18. //建立一个TCP套接口   
  19. if((sockfd = socket(AF_INET,SOCK_STREAM,0))==-1) {  
  20. perror("socket");  
  21. exit(1);  
  22. }  
  23. //初始化结构体,连接到服务器的2323端口   
  24. their_addr.sin_family = AF_INET;  
  25. their_addr.sin_port = htons(2323);  
  26. their_addr.sin_addr = *((struct in_addr *)he->h_addr);  
  27. bzero(&(their_addr.sin_zero),8);  
  28. //和服务器建立连接   
  29. if(connect(sockfd,(struct sockaddr *)&their_addr,  
  30. sizeof(struct sockaddr))  
  31. ==-1){  
  32. perror("connect");  
  33. exit(1);  
  34. }  
  35. //向服务器发送字符串"hello!"   
  36. if(send(sockfd,"hello!",6,0)==-1) {  
  37. perror("send");  
  38. exit(1);  
  39. }  
  40. //接受从服务器返回的信息   
  41. if((numbytes = recv(sockfd,buf,100,0))==-1) {  
  42. perror("recv");  
  43. exit(1);  
  44. }  
  45. buf[numbytes] = '';  
  46. printf("result:%s",buf);  
  47. close(sockfd);  
  48. return 0;  
  49. }  
  50. --------------------------------------------------------------------  
  51. /* server.c */  
  52. #include <stdio.h>   
  53. #include <stdlib.h>   
  54. #include <errno.h>   
  55. #include <string.h>   
  56. #include <sys/types.h>   
  57. #include <netinet/in.h>   
  58. #include <sys/socket.h>   
  59. #include <sys/wait.h>   
  60. main() {  
  61. int sockfd,new_fd;  
  62. struct sockaddr_in my_addr;  
  63. struct sockaddr_in their_addr;  
  64. int sin_size;  
  65. //建立TCP套接口   
  66. if((sockfd = socket(AF_INET,SOCK_STREAM,0))==-1) {  
  67. perror("socket");  
  68. exit(1);  
  69. }  
  70. //初始化结构体,并绑定2323端口   
  71. my_addr.sin_family = AF_INET;  
  72. my_addr.sin_port = htons(2323);  
  73. my_addr.sin_addr.s_addr = INADDR_ANY;  
  74. bzero(&(my_addr.sin_zero),8);  
  75. //绑定套接口   
  76. if(bind(sockfd,(struct sockaddr *)&my_addr,sizeof(struct  
  77.  sockaddr))==-1)  
  78. {  
  79. perror("bind");  
  80. exit(1);  
  81. }  
  82. //创建监听套接口   
  83. if(listen(sockfd,10)==-1) {  
  84. perror("listen");  
  85. exit(1);  
  86. }  
  87. //等待连接   
  88. while(1) {  
  89. sin_size = sizeof(struct sockaddr_in);  
  90. perror("server is run");  
  91. //如果建立连接,将产生一个全新的套接字   
  92. if((new_fd = accept(sockfd,(struct sockaddr *)  
  93. &their_addr,&sin_size))==-1)  
  94. {  
  95. perror("accept");  
  96. exit(1);  
  97. }  
  98. //生成一个子进程来完成和客户端的会话,父进程继续监听   
  99. if(!fork()) {  
  100. //读取客户端发来的信息   
  101. if((numbytes = recv(new_fd,buff,strlen(buff),0))==-1)  
  102. {  
  103. perror("recv");  
  104. exit(1);  
  105. }  
  106. printf("%s",buff);  
  107. //将从客户端接收到的信息再发回客户端   
  108. if(send(new_fd,buff,strlen(buff),0)==-1)  
  109. perror("send");  
  110. close(new_fd);  
  111. exit(0);  
  112. }  
  113. close(new_fd);  
  114. }  
  115. close(sockfd);  
  116. }  

网络编程中设计并发服务器,使用多进程 与 多线程 ,请问有什么区别?

linux C网络编程基本流程_第1张图片


文章转自:http://blog.csdn.net/randyjiawenjie/article/details/6895979

你可能感兴趣的:(linux C网络编程基本流程)