Linux下网络编程实验

 (1)TCP程序设计

服务器端tcp_server.c

 

 
 
  1. #include <stdlib.h>  
  2. #include <stdio.h>  
  3. #include <errno.h>  
  4. #include <string.h>  
  5. #include <netdb.h>  
  6. #include <sys/types.h>  
  7. #include <netinet/in.h>  
  8. #include <sys/socket.h>  
  9.  
  10. #define portnumber 3333 
  11.  
  12. int main(int argc, char *argv[])  
  13. {  
  14.     int sockfd,new_fd;  
  15.     struct sockaddr_in server_addr;  
  16.     struct sockaddr_in client_addr;  
  17.     int sin_size;  
  18.     int nbytes; 
  19.     char buffer[1024]; 
  20.      
  21.  
  22.     /* 服务器端开始建立sockfd描述符 */  
  23.     if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1) // AF_INET:IPV4;SOCK_STREAM:TCP 
  24.     {  
  25.         fprintf(stderr,"Socket error:%s\n\a",strerror(errno));  
  26.         exit(1);  
  27.     }  
  28.  
  29.     /* 服务器端填充 sockaddr结构 */  
  30.     bzero(&server_addr,sizeof(struct sockaddr_in)); // 初始化,置0 
  31.     server_addr.sin_family=AF_INET;                 // Internet 
  32.     server_addr.sin_addr.s_addr=htonl(INADDR_ANY);  // (将本机器上的long数据转化为网络上的long数据)和任何主机通信  //INADDR_ANY 表示可以接收任意IP地址的数据,即绑定到所有的IP 
  33.     //server_addr.sin_addr.s_addr=inet_addr("192.168.1.1");  //用于绑定到一个固定IP,inet_addr用于把数字加格式的ip转化为整形ip 
  34.     server_addr.sin_port=htons(portnumber);         // (将本机器上的short数据转化为网络上的short数据)端口号 
  35.      
  36.     /* 捆绑sockfd描述符到IP地址 */  
  37.     if(bind(sockfd,(struct sockaddr *)(&server_addr),sizeof(struct sockaddr))==-1)  
  38.     {  
  39.         fprintf(stderr,"Bind error:%s\n\a",strerror(errno));  
  40.         exit(1);  
  41.     }  
  42.  
  43.     /* 设置允许连接的最大客户端数 */  
  44.     if(listen(sockfd,5)==-1)  
  45.     {  
  46.         fprintf(stderr,"Listen error:%s\n\a",strerror(errno));  
  47.         exit(1);  
  48.     }  
  49.  
  50.     while(1)  
  51.     {  
  52.         /* 服务器阻塞,直到客户程序建立连接 */  
  53.         sin_size=sizeof(struct sockaddr_in);  
  54.         if((new_fd=accept(sockfd,(struct sockaddr *)(&client_addr),&sin_size))==-1)  
  55.         {  
  56.             fprintf(stderr,"Accept error:%s\n\a",strerror(errno));  
  57.             exit(1);  
  58.         }  
  59.         fprintf(stderr,"Server get connection from %s\n",inet_ntoa(client_addr.sin_addr)); // 将网络地址转换成.字符串 
  60.          
  61.         if((nbytes=read(new_fd,buffer,1024))==-1)  
  62.         {  
  63.             fprintf(stderr,"Read Error:%s\n",strerror(errno));  
  64.             exit(1);  
  65.         }        
  66.         buffer[nbytes]='\0'
  67.         printf("Server received %s\n",buffer); 
  68.          
  69.         /* 这个通讯已经结束 */  
  70.         close(new_fd);  
  71.         /* 循环下一个 */  
  72.     }  
  73.  
  74.     /* 结束通讯 */  
  75.     close(sockfd);  
  76.     exit(0);  
  77. }  
  78.  

客户端tcp_client.c

 

 
 
  1. #include <stdlib.h>  
  2. #include <stdio.h>  
  3. #include <errno.h>  
  4. #include <string.h>  
  5. #include <netdb.h>  
  6. #include <sys/types.h>  
  7. #include <netinet/in.h>  
  8. #include <sys/socket.h>  
  9.  
  10. #define portnumber 3333 
  11.  
  12. int main(int argc, char *argv[])  
  13. {  
  14.     int sockfd;  
  15.     char buffer[1024];  
  16.     struct sockaddr_in server_addr;  
  17.     struct hostent *host;  
  18.  
  19.         /* 使用hostname查询host 名字 */ 
  20.     if(argc!=2)  
  21.     {  
  22.         fprintf(stderr,"Usage:%s hostname \a\n",argv[0]);  
  23.         exit(1);  
  24.     }  
  25.  
  26.     if((host=gethostbyname(argv[1]))==NULL)  
  27.     {  
  28.         fprintf(stderr,"Gethostname error\n");  
  29.         exit(1);  
  30.     }  
  31.  
  32.     /* 客户程序开始建立 sockfd描述符 */  
  33.     if((sockfd=socket(AF_INET,SOCK_STREAM,0))==-1) // AF_INET:Internet;SOCK_STREAM:TCP 
  34.     {  
  35.         fprintf(stderr,"Socket Error:%s\a\n",strerror(errno));  
  36.         exit(1);  
  37.     }  
  38.  
  39.     /* 客户程序填充服务端的资料 */  
  40.     bzero(&server_addr,sizeof(server_addr)); // 初始化,置0 
  41.     server_addr.sin_family=AF_INET;          // IPV4 
  42.     server_addr.sin_port=htons(portnumber);  // (将本机器上的short数据转化为网络上的short数据)端口号 
  43.     server_addr.sin_addr=*((struct in_addr *)host->h_addr); // IP地址 
  44.      
  45.     /* 客户程序发起连接请求 */  
  46.     if(connect(sockfd,(struct sockaddr *)(&server_addr),sizeof(struct sockaddr))==-1)  
  47.     {  
  48.         fprintf(stderr,"Connect Error:%s\a\n",strerror(errno));  
  49.         exit(1);  
  50.     }  
  51.  
  52.     /* 连接成功了 */  
  53.     printf("Please input char:\n"); 
  54.      
  55.     /* 发送数据 */ 
  56.     fgets(buffer,1024,stdin);  
  57.     write(sockfd,buffer,strlen(buffer));  
  58.  
  59.     /* 结束通讯 */  
  60.     close(sockfd);  
  61.     exit(0);  
  62. }  

(2)UDP程序设计

服务器端udp_server.c

 

 
 
  1. #include <stdlib.h> 
  2. #include <stdio.h> 
  3. #include <errno.h> 
  4. #include <string.h> 
  5. #include <unistd.h> 
  6. #include <netdb.h> 
  7. #include <sys/socket.h> 
  8. #include <netinet/in.h> 
  9. #include <sys/types.h> 
  10. #include <arpa/inet.h> 
  11.  
  12. #define SERVER_PORT 8888  
  13. #define MAX_MSG_SIZE 1024  
  14.  
  15. void udps_respon(int sockfd)  
  16. {  
  17.     struct sockaddr_in addr;  
  18.     int addrlen,n;  
  19.     char msg[MAX_MSG_SIZE];  
  20.  
  21.     while(1)  
  22.     {   /* 从网络上读,并写到网络上 */  
  23.         bzero(msg,sizeof(msg)); // 初始化,清零 
  24.         addrlen = sizeof(struct sockaddr);  
  25.         n=recvfrom(sockfd,msg,MAX_MSG_SIZE,0,(struct sockaddr*)&addr,&addrlen); // 从客户端接收消息 
  26.         msg[n]=0;  
  27.         /* 显示服务端已经收到了信息 */  
  28.         fprintf(stdout,"Server have received %s",msg); // 显示消息 
  29.     }  
  30. }  
  31.  
  32. int main(void)  
  33. {  
  34.     int sockfd;  
  35.     struct sockaddr_in addr;  
  36.  
  37.     /* 服务器端开始建立socket描述符 */  
  38.     sockfd=socket(AF_INET,SOCK_DGRAM,0);  
  39.     if(sockfd<0)  
  40.     {  
  41.         fprintf(stderr,"Socket Error:%s\n",strerror(errno));  
  42.         exit(1);  
  43.     }  
  44.  
  45.     /* 服务器端填充 sockaddr结构 */  
  46.     bzero(&addr,sizeof(struct sockaddr_in));  
  47.     addr.sin_family=AF_INET;  
  48.     addr.sin_addr.s_addr=htonl(INADDR_ANY);  
  49.     addr.sin_port=htons(SERVER_PORT);  
  50.  
  51.     /* 捆绑sockfd描述符 */  
  52.     if(bind(sockfd,(struct sockaddr *)&addr,sizeof(struct sockaddr_in))<0)  
  53.     {  
  54.         fprintf(stderr,"Bind Error:%s\n",strerror(errno));  
  55.         exit(1);  
  56.     }  
  57.  
  58.     udps_respon(sockfd); // 进行读写操作 
  59.     close(sockfd);  
  60. }  

客户端udp_client.c

 

 
 
  1. #include <stdlib.h> 
  2. #include <stdio.h> 
  3. #include <errno.h> 
  4. #include <string.h> 
  5. #include <unistd.h> 
  6. #include <netdb.h> 
  7. #include <sys/socket.h> 
  8. #include <netinet/in.h> 
  9. #include <sys/types.h> 
  10. #include <arpa/inet.h> 
  11.  
  12. #define SERVER_PORT 8888  
  13. #define MAX_BUF_SIZE 1024  
  14.  
  15. void udpc_requ(int sockfd,const struct sockaddr_in *addr,int len)  
  16. {  
  17.     char buffer[MAX_BUF_SIZE];  
  18.     int n;  
  19.     while(1)  
  20.     {   /* 从键盘读入,写到服务端 */  
  21.         printf("Please input char:\n"); 
  22.         fgets(buffer,MAX_BUF_SIZE,stdin);  
  23.         sendto(sockfd,buffer,strlen(buffer),0,addr,len);  
  24.         bzero(buffer,MAX_BUF_SIZE);  
  25.     }  
  26. }  
  27.  
  28. int main(int argc,char **argv)  
  29. {  
  30.     int sockfd;  
  31.     struct sockaddr_in addr;  
  32.  
  33.     if(argc!=2)  
  34.     {  
  35.         fprintf(stderr,"Usage:%s server_ip\n",argv[0]);  
  36.         exit(1);  
  37.     } 
  38.  
  39.     /* 建立 sockfd描述符 */  
  40.     sockfd=socket(AF_INET,SOCK_DGRAM,0);  
  41.     if(sockfd<0)  
  42.     {  
  43.         fprintf(stderr,"Socket Error:%s\n",strerror(errno));  
  44.         exit(1);  
  45.     }  
  46.  
  47.     /* 填充服务端的资料 */  
  48.     bzero(&addr,sizeof(struct sockaddr_in));  
  49.     addr.sin_family=AF_INET;  
  50.     addr.sin_port=htons(SERVER_PORT); 
  51.     if(inet_aton(argv[1],&addr.sin_addr)<0)  /*inet_aton函数用于把字符串型的IP地址转化成网络2进制数字*/  
  52.     {  
  53.         fprintf(stderr,"Ip error:%s\n",strerror(errno));  
  54.         exit(1);  
  55.     }  
  56.  
  57.     udpc_requ(sockfd,&addr,sizeof(struct sockaddr_in)); // 进行读写操作 
  58.     close(sockfd);  
  59. }  

(3)并发服务器设计(TCP需建立连接,故为循环服务器,可通过创建一个子进程fork()来实现;UDP可并发)

服务器端tcp_server_fork.c,客户端同上

 
 
  1. #include <stdlib.h>  
  2. #include <stdio.h>  
  3. #include <errno.h>  
  4. #include <string.h>  
  5. #include <netdb.h>  
  6. #include <sys/types.h>  
  7. #include <netinet/in.h>  
  8. #include <sys/socket.h>  
  9.  
  10. #define MY_PORT 3333 
  11.  
  12.  
  13.  
  14. int main(int argc ,char **argv) 
  15.  
  16.  
  17.  int listen_fd,accept_fd; 
  18.  
  19.  struct sockaddr_in     client_addr; 
  20.  
  21.  int n; 
  22.  
  23.   
  24.  
  25.  if((listen_fd=socket(AF_INET,SOCK_STREAM,0))<0) 
  26.  
  27.   { 
  28.  
  29.         printf("Socket Error:%s\n\a",strerror(errno)); 
  30.  
  31.         exit(1); 
  32.  
  33.   } 
  34.  
  35.   
  36.  
  37.  bzero(&client_addr,sizeof(struct sockaddr_in)); 
  38.  
  39.  client_addr.sin_family=AF_INET; 
  40.  
  41.  client_addr.sin_port=htons(MY_PORT); 
  42.  
  43.  client_addr.sin_addr.s_addr=htonl(INADDR_ANY); 
  44.  
  45.  n=1; 
  46.  
  47.  /* 如果服务器终止后,服务器可以第二次快速启动而不用等待一段时间  */ 
  48.  
  49.  setsockopt(listen_fd,SOL_SOCKET,SO_REUSEADDR,&n,sizeof(int)); 
  50.  
  51.  if(bind(listen_fd,(struct sockaddr *)&client_addr,sizeof(client_addr))<0) 
  52.  
  53.   { 
  54.  
  55.         printf("Bind Error:%s\n\a",strerror(errno)); 
  56.  
  57.         exit(1); 
  58.  
  59.   } 
  60.  
  61.   listen(listen_fd,5); 
  62.  
  63.   while(1) 
  64.  
  65.   { 
  66.  
  67.    accept_fd=accept(listen_fd,NULL,NULL); 
  68.  
  69.    if((accept_fd<0)&&(errno==EINTR)) 
  70.  
  71.           continue
  72.  
  73.    else if(accept_fd<0) 
  74.  
  75.     { 
  76.  
  77.         printf("Accept Error:%s\n\a",strerror(errno)); 
  78.  
  79.         continue
  80.  
  81.     } 
  82.  
  83.   if((n=fork())==0) 
  84.  
  85.    { 
  86.  
  87.         /* 子进程处理客户端的连接 */ 
  88.  
  89.         char buffer[1024]; 
  90.  
  91.  
  92.  
  93.         close(listen_fd); 
  94.  
  95.         n=read(accept_fd,buffer,1024); 
  96.  
  97.         write(accept_fd,buffer,n); 
  98.  
  99.         close(accept_fd); 
  100.  
  101.         exit(0); 
  102.  
  103.    } 
  104.  
  105.    else if(n<0) 
  106.  
  107.         printf("Fork Error:%s\n\a",strerror(errno)); 
  108.  
  109.    close(accept_fd); 
  110.  
  111.   } 
  112.  
  113. }  

 

本文出自 “Mr~钟” 博客,请务必保留此出处http://6386296.blog.51cto.com/6376296/1121626

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值