1. 程式人生 > >Linux網路程式設計案例分析

Linux網路程式設計案例分析

本程式碼來自於博主:輝夜星辰 

本篇主要對執行程式碼中出現的問題進行分析,程式碼本身的內容後續展開討論。

  • 伺服器端程式碼
  1 /*
2 Linux網路程式設計之TCP程式設計,伺服器端讀資料 3 socket函式之後,返回值serfd,作為後面所有網路程式設計函式的第一個引數 4 */ 5 #include <stdio.h> 6 #include <stdlib.h> 7 #include <string.h> 8 #include <strings.h> 9 #include <unistd.h> 10
#include <sys/types.h> 11 #include <sys/socket.h> 12 #include <arpa/inet.h> 13 #include <netinet/in.h> 14 15 #define SER_PORT 8888 //埠號在5001-65535之間 16 #define SER_IP "192.168.7.115" 17 18 /* 19 1.sockfd = socket(int socket_family, int socket_type, int protocol); 20 21 2.int bind(int sockfd, const struct sockaddr *addr,
22 socklen_t addrlen);. 23 3.//通用地址結構 24 struct sockaddr { 25 sa_family_t sa_family;//地址族,AF_xxx 26 char sa_data[14];//14位元組協議地址 27 } 28 29 4.//Internet協議地址結構(一般用這個) 30 struct sockaddr_in 31 { 32 u_short sin_family; // 地址族, AF_INET,2 bytes
33 u_short sin_port; // 埠,2 bytes 34 struct in_addr sin_addr; // IPV4地址,4 bytes 35 char sin_zero[8]; // 8 bytes unused,作為填充 36 }; 37 38 5.//IPv4地址結構 39 // internet address 40 struct in_addr 41 { 42 in_addr_t s_addr; // u32 network address 43 }; 44 45 6.//位元組序轉換函式 46 主機位元組序到網路位元組序 47 u_long htonl (u_long hostlong); 轉四個位元組的 48 u_short htons (u_short short); 轉兩個位元組的 49 50 網路位元組序到主機位元組序 51 u_long ntohl (u_long hostlong);轉四個位元組的 52 u_short ntohs (u_short short);轉兩個位元組的 53 54 7.//IP地址的轉換 55 inet_addr( ) 56 將strptr所指的字串轉換成32位的網路位元組序二進位制值,返回轉換後的地址。 57 int_addr_t inet_addr(const char *strptr); 58 59 60 8.int listen(int sockfd, int backlog); 61 62 9.int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen); 63 64 */ 65 66 //出錯處理函式 67 void sys_error(char *ch) 68 { 69 perror(ch); 70 exit(1); 71 } 72 73 int main(int argc,char *argv[]) 74 { 75 int serfd,ret; 76 char buf[128]; 77 //1.建立流式套接字 78 serfd = socket(AF_INET,SOCK_STREAM,0);//返回套接字描述符 79 if(serfd < 0) 80 sys_error("socket failed"); 81 82 //2.繫結本地ip和埠 83 struct sockaddr_in ser; 84 bzero(&ser,sizeof(ser));//清空結構體快取 85 ser.sin_family = AF_INET;//選擇IPV4 86 ser.sin_port = htons(SER_PORT);//填充埠 87 ser.sin_addr.s_addr = inet_addr(SER_IP);//填充IP 88 89 ret=bind(serfd,(struct sockaddr *)&ser,sizeof(ser));//強制型別轉換 90 if(ret < 0) 91 sys_error("bind failed"); 92 93 //3.監聽 94 ret=listen(serfd,5);//請求佇列中允許的最大請求數,一般為5 95 if(ret < 0) 96 sys_error("listen failed"); 97 printf("listent ok\n"); 98 99 //4.接收 100 struct sockaddr_in self; 101 bzero(&self,sizeof(self));//清空結構體快取 102 int len=sizeof(self); 103 int newfd;//返回套接字描述符 104 newfd = accept(serfd,(struct sockaddr *)&self,&len); 105 if(newfd < 0) 106 sys_error("accept failed"); 107 108 //5.讀取資料 109 while(1) 110 { 111 bzero(buf,128);//清空buf快取 112 ret = read(newfd,buf,128);//從客戶端讀取資料 113 if(ret<0) 114 sys_error("read failed"); 115 else if(ret == 0) 116 { 117 fprintf(stdout,"Bye-Bye\n"); 118 break; 119 } 120 else 121 printf("buf from client:%s",buf); 122 } 123 124 //6.關閉(2個套接字描述符) 125 126 close(serfd); 127 close(newfd); 128 return 0; 129 } 130 131

 

  • 客戶端程式碼
  1 /*
  2 Linux網路程式設計之TCP程式設計,客戶端寫資料
  3 */
  4 #include <stdio.h>
  5 #include <stdlib.h>
  6 #include <string.h>
  7 #include <strings.h>
  8 #include <unistd.h>
  9 #include <sys/types.h>
 10 #include <sys/socket.h>
 11 #include <arpa/inet.h>
 12 #include <netinet/in.h>
 13 
 14 #define SER_PORT 8888 //埠號在5001-65535之間
 15 #define SER_IP "192.168.7.115" //寫資料物件的ip
 16 
 17 /*
 18  1.sockfd = socket(int socket_family, int socket_type, int protocol);
 19 
 20  2.//通用地址結構         
 21     struct sockaddr {
 22                sa_family_t sa_family;//地址族,AF_xxx
 23                char        sa_data[14];//14位元組協議地址
 24            }
 25 
 26  3.//Internet協議地址結構(一般用這個)
 27     struct sockaddr_in
 28       {           
 29            u_short sin_family;      // 地址族, AF_INET,2 bytes
 30            u_short sin_port;      // 埠,2 bytes
 31            struct in_addr sin_addr;  // IPV4地址,4 bytes  
 32            char sin_zero[8];        // 8 bytes unused,作為填充
 33       };
 34 
 35  4.//IP地址的轉換
 36     inet_addr( )
 37     將strptr所指的字串轉換成32位的網路位元組序二進位制值,返回轉換後的地址。
 38     int_addr_t   inet_addr(const char *strptr);
 39 
 40  5.int connect(int sockfd, const struct sockaddr *addr,
 41                    socklen_t addrlen);
 42 
 43  6.ssize_t send(int sockfd, const void *buf, size_t len, int flags);                   
 44 */
 45 
 46 void sys_error(char *ch)
 47 {
 48     perror(ch);
 49     exit(1);
 50 }
 51 
 52 int main(int argc,char *argv[])
 53 {
 54     int clifd,ret;
 55     char buf[128];
 56 
 57     //1.建立流式套接字
 58     clifd = socket(AF_INET,SOCK_STREAM,0);
 59     if(clifd < 0)
 60         sys_error("socket failed");
 61 
 62     //2.主動發起連線
 63     struct sockaddr_in cli;
 64     bzero(&cli,sizeof(cli));//清空結構體快取
 65     cli.sin_family = AF_INET;//選擇IPV4
 66     cli.sin_port = htons(SER_PORT);//填充埠
 67     cli.sin_addr.s_addr = inet_addr(SER_IP);//填充IP
 68 
 69     ret = connect(clifd,(struct sockaddr *)&cli,sizeof(cli));
 70     if(ret < 0)
 71         sys_error("connect failed");
 72 
 73     printf("connect ok\n");
 74 
 75     //3.寫資料
 76     while(1)
 77     {
 78         bzero(buf,128);
 79         fprintf(stderr,"please input:");
 80         fgets(buf,128,stdin);
 81         //寫資料到伺服器
 82         if(write(clifd,buf,strlen(buf)) < 0)
 83             sys_error("write failed");
 84         if(!strncmp(buf,"quit",4))
 85             break;
 86     }
 87 
 88     //4.關閉
 89     close(clifd);
 90 
 91     return 0;
 92 }
 93

 

鑑於自己剛剛起步學習Linux網路程式設計,看完書後一般是現在網上找一些例項程式碼進行測試,便於理解。

我網上找了些程式碼例項,經常會遇到出錯情況,而上述程式碼執行沒有錯誤,但是需要注意幾點:

 

下面程式碼(伺服器程式碼與客戶端程式碼均需要修改)處需要進行修改,此處的ip應該填寫你自己電腦的ip,在ubuntu系統“終端”中執行"ifconfig",查到自己本機的ip,然後填寫進去。

 
 1 #define SER_IP "192.168.7.115" //寫資料物件的ip
 

關於程式碼的執行問題,當初也是困惑了我良久,若你已經安裝了gcc,那麼在終端下執行下面程式碼,前提是你得知道自己的程式碼放入的路徑在哪兒:

 1 gcc filename.c -o filename //此處filename表示自己的C檔名,執行完之後接著執行第二步
1 gcc ./filename //回車後程序就開始執行了

 

第三個注意點就是伺服器程式需要使用一個終端進行執行,客戶端程式需要使用一個終端進行執行,伺服器程式需要先執行。

伺服器程式啟動,開始監聽:

 

客戶端程式連線伺服器完成,並提示輸入:

 

 客戶端輸入完成,回車後,伺服器程式執行的終端上會立即顯示客戶端傳送的資訊.(左為客戶端程式,右為伺服器程式)

 

 上述可以不聽輸入,需要終止時,即在客戶端程式中輸入quit,伺服器端顯示客戶端傳送的“quit”,另起一行彈出“Bye-Bye“結束程式。

 

 

 上述是對程式的簡要分析,現在看起來簡單,但是第一次執行還是遇到了一定的困難,接下來集中精力好好研究程式碼本身的內容啦。