服務端
/************************************************************************************************************************ 1、int socket(int family,int type,int protocol) family: 指定使用的協議簇:AF_INET(IPv4) AF_INET6(IPv6) AF_LOCAL(UNIX協議) AF_ROUTE(路由套接字) AF_KEY(秘鑰套接字) type: 指定使用的套接字的類型:SOCK_STREAM(字節流套接字) SOCK_DGRAM protocol: 如果套接字類型不是原始套接字,那么這個參數就為0 2、int bind(int sockfd, struct sockaddr *myaddr, int addrlen) sockfd: socket函數返回的套接字描述符 myaddr: 是指向本地IP地址的結構體指針 myaddrlen: 結構長度 struct sockaddr{ unsigned short sa_family; //通信協議類型族AF_xx char sa_data[14]; //14字節協議地址,包含該socket的IP地址和端口號 }; struct sockaddr_in{ short int sin_family; //通信協議類型族 unsigned short int sin_port; //端口號 struct in_addr sin_addr; //IP地址 unsigned char si_zero[8]; //填充0以保持與sockaddr結構的長度相同 }; 3、int connect(int sockfd,const struct sockaddr *serv_addr,socklen_t addrlen) sockfd: socket函數返回套接字描述符 serv_addr: 服務器IP地址結構指針 addrlen: 結構體指針的長度 4、int listen(int sockfd, int backlog) sockfd: socket函數綁定bind后套接字描述符 backlog: 設置可連接客戶端的最大連接個數,當有多個客戶端向服務器請求時,收到此值的影響。默認值20 5、int accept(int sockfd,struct sockaddr *cliaddr,socklen_t *addrlen) sockfd: socket函數經過listen后套接字描述符 cliaddr: 客戶端套接字接口地址結構 addrlen: 客戶端地址結構長度 6、int send(int sockfd, const void *msg,int len,int flags) 7、int recv(int sockfd, void *buf,int len,unsigned int flags) sockfd: socket函數的套接字描述符 msg: 發送數據的指針 buf: 存放接收數據的緩沖區 len: 數據的長度,把flags設置為0 *************************************************************************************************************************/
讀函數read
ssize_t read(int fd,void *buf,size_t nbyte)
read函數是負責從fd中讀取內容.成功時,read返回實際所讀的字節數,如果返回的值是0,表示已經讀到文件的結束了.
小於0表示出現了錯誤.如果錯誤為EINTR說明讀是由中斷引起的, 如果是ECONNREST表示網絡連接出了問題.參數nbyte是請求讀取的字節數,讀上來的數據保存在緩沖區buf中,同時文件的當前讀寫位置向后移。注意這個讀寫位置和使用C標准I/O庫時的讀寫位置有可能不同,這個讀寫位置是記在內核中的
寫函數write
ssize_t write(int fd,const void *buf,size_t nbytes)
write函數將buf中的nbytes字節內容寫入文件描述符fd.成功時返回寫的字節數.失敗時返回-1. 並設置errno變量. 在網絡程序中,當我們向套接字文件描述符寫時有倆種可能.
1)write的返回值大於0,表示寫了部分或者是全部的數據.
2)返回的值小於0,此時出現了錯誤.我們要根據錯誤類型來處理. 如果錯誤為EINTR表示在寫的時候出現了中斷錯誤.
如果為EPIPE表示網絡連接出現了問題(對方已經關閉了連接).
1)write的返回值大於0,表示寫了部分或者是全部的數據.
2)返回的值小於0,此時出現了錯誤.我們要根據錯誤類型來處理. 如果錯誤為EINTR表示在寫的時候出現了中斷錯誤.
如果為EPIPE表示網絡連接出現了問題(對方已經關閉了連接).
#include<stdio.h> #include<stdlib.h> #include<errno.h> #include<string.h> #include<sys/types.h> #include<netinet/in.h> #include<sys/socket.h> #include<sys/wait.h> #define PORT 1500//端口號 #define BACKLOG 5/*最大監聽數*/ int main(){ int sockfd,new_fd;/*socket句柄和建立連接后的句柄*/ struct sockaddr_in my_addr;/*本方地址信息結構體,下面有具體的屬性賦值*/ struct sockaddr_in their_addr;/*對方地址信息*/ int sin_size; sockfd=socket(AF_INET,SOCK_STREAM,0);//建立socket if(sockfd==-1){ printf("socket failed:%d",errno); return -1; } my_addr.sin_family=AF_INET;/*該屬性表示接收本機或其他機器傳輸*/ my_addr.sin_port=htons(PORT);/*端口號*/ my_addr.sin_addr.s_addr=htonl(INADDR_ANY);/*IP,括號內容表示本機IP*/ bzero(&(my_addr.sin_zero),8);/*將其他屬性置0*/ if(bind(sockfd,(struct sockaddr*)&my_addr,sizeof(struct sockaddr))<0){//綁定地址結構體和socket printf("bind error"); return -1; } listen(sockfd,BACKLOG);//開啟監聽 ,第二個參數是最大監聽數 while(1){ sin_size=sizeof(struct sockaddr_in); new_fd=accept(sockfd,(struct sockaddr*)&their_addr,&sin_size);//在這里阻塞知道接收到消息,參數分別是socket句柄,接收到的地址信息以及大小 if(new_fd==-1){ printf("receive failed"); } else{ printf("receive success"); send(new_fd,"Hello World!",12,0);//發送內容,參數分別是連接句柄,內容,大小,其他信息(設為0即可) } } return 0; }
客戶端
#include<stdio.h> #include<stdlib.h> #include<errno.h> #include<string.h> #include<sys/types.h> #include<netinet/in.h> #include<sys/socket.h> #include<sys/wait.h> #define DEST_PORT 1500//目標地址端口號 #define DEST_IP "127.0.0.1"/*目標地址IP,這里設為本機*/ #define MAX_DATA 100//接收到的數據最大程度 int main(){ int sockfd,new_fd;/*cocket句柄和接受到連接后的句柄 */ struct sockaddr_in dest_addr;/*目標地址信息*/ char buf[MAX_DATA];//儲存接收數據 sockfd=socket(AF_INET,SOCK_STREAM,0);/*建立socket*/ if(sockfd==-1){ printf("socket failed:%d",errno); } //參數意義見上面服務器端 dest_addr.sin_family=AF_INET; dest_addr.sin_port=htons(DEST_PORT); dest_addr.sin_addr.s_addr=inet_addr(DEST_IP); bzero(&(dest_addr.sin_zero),8); if(connect(sockfd,(struct sockaddr*)&dest_addr,sizeof(struct sockaddr))==-1){//連接方法,傳入句柄,目標地址和大小 printf("connect failed:%d",errno);//失敗時可以打印errno } else{ printf("connect success"); recv(sockfd,buf,MAX_DATA,0);//將接收數據打入buf,參數分別是句柄,儲存處,最大長度,其他信息(設為0即可)。 printf("Received:%s",buf); } close(sockfd);//關閉socket return 0; }
