recv和send函數:
#include<sys/socket.h>
ssize_t recv(int sockfd, void *buff, size_t nbytes, int flags);
ssize_t recv(int sockfd, const void *buff, size_t nbytes, int flags);
flags的值中 MSG_OOB和MSG_PEEK比較重要。
read和recv函數的區別在於:
read函數讀取緩沖區的數據之后,會將緩沖區的數據刪除,而recv不會刪除緩沖區的數據。
因此,可以將falgs設置為MSG_PEEK,在此模式下,先查看發送過來的字符的個數,然后用read函數讀取數據。
例子服務端代碼如下:
#include<stdio.h> #include<stdlib.h> #include<errno.h> #include<string.h> #include<sys/types.h> #include<sys/socket.h> #include<sys/un.h> #include<sys/wait.h> //*進程用的頭文件*/ #include<netinet/in.h> #include<arpa/inet.h> #include<unistd.h> #define MAXLINE 1024 //通信內容的最大長度 ssize_t readn(int fd, void *buf, size_t count) { ssize_t nleft=count; ssize_t nread; char *charbuf=(char*) buf; while(nleft>0) { nread=read(fd,charbuf,nleft); if(nread<0) { if(errno==EINTR) continue; return -1; } else if(nread==0) return count-nleft; charbuf +=nread; nleft=count-nread; } return count; } ssize_t writen(int fd, const void *buf, size_t count) { ssize_t nleft=count; ssize_t nwrite; char *charbuf=(char*) buf; while(nleft>0) { nwrite=write(fd,charbuf,nleft); if(nwrite<0) { if(errno==EINTR) continue; return -1; } else if(nwrite==0) return count-nleft; charbuf +=nwrite; nleft=count-nwrite; } return count; } ssize_t recv_peek(int sockfd, void *buf, size_t len) { int ret; while(1) { ret=recv(sockfd,buf,len,MSG_PEEK); if(ret==-1&& errno==EINTR) continue; return ret; } } ssize_t readline(int sockfd, void *buf, size_t len) { ssize_t nleft=len,nread; int ret; char* bufchar=buf; while(1) { ret=recv_peek(sockfd,bufchar,len); if(ret<0||ret==0) return ret; nread=ret; int i; for(i=0;i<nread;i++) { if(bufchar[i]=='\n') { ret=readn(sockfd,bufchar,i+1); if(ret!=i+1) exit(EXIT_FAILURE); return ret; } } if(nread>nleft) exit(EXIT_FAILURE); nleft-=nread; ret=readn(sockfd,bufchar,nread); if(ret!=nread) exit(EXIT_FAILURE); bufchar+=nread; } return -1; } int main() { int sock_fd,new_fd;//sock_fd用於監聽,new_fd用於連接 struct sockaddr_in srv_addr;//服務器的地址信息 struct sockaddr_in client_addr;//客戶機的地址信息 int size; //地址結構數據的長度 pid_t pid; //子進程id ssize_t n,ret; char buf[MAXLINE]; //用於存放通信的內容 char sendbuf[1024],recvbuf[1024]; memset(sendbuf,0,sizeof(sendbuf)); memset(recvbuf,0,sizeof(recvbuf)); /*創建套接字*/ sock_fd=socket(AF_INET,SOCK_STREAM,0);//采用IPv4協議 if(sock_fd==-1) { perror("creat socket failed"); exit(1); } /*服務器地址參數*/ srv_addr.sin_family=AF_INET; srv_addr.sin_port=htons(3490); srv_addr.sin_addr.s_addr=htonl(INADDR_ANY); bzero(&srv_addr.sin_zero,sizeof(struct sockaddr_in));//bzero位清零函數,將sin_zero清零,sin_zero為填充字段,必須全部為零 int on=1; //表示開啟reuseaddr if(setsockopt(sock_fd,SOL_SOCKET,SO_REUSEADDR,&on,sizeof(on))<0) //打開地址、端口重用 perror("setsockopt"); /*綁定地址和端口*/ if(bind(sock_fd,(struct sockaddr*)&srv_addr,sizeof(struct sockaddr))==-1) { perror("bind failed"); exit(1); } /*連接到服務器 if(connect(sock_fd,(struct sockaddr*)&srv_addr,sizeof(sock_fd))==-1) { perror("bind failed"); exit(1); }*/ /*設置監聽模式,等待客戶機的監聽*/ if((listen(sock_fd,5))==-1) { perror("listen failed"); exit(1); } /*接受連接,采用非阻塞是的模式調用accep*/ //while(1) //{ size=sizeof(struct sockaddr_in); new_fd=accept(sock_fd,(struct sockaddr*)&client_addr,&size); if(new_fd==-1) { perror("accept failed"); //continue;//restart accept when EINTR } printf("server:got connection from IP= %s prot= %d \n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));//連接成功,打印客戶機IP地址和端口號 /*char *inet_nota(struct sockaddr_in in); 頭文件: arpa/inet.h Winsock2.h 參數: 一個網絡上的IP地址 返回值: 如果正確,返回一個字符指針,指向一塊存儲着點分格式IP地址的靜態緩沖區(同一線程內共享此內存);錯誤,返回NULL。 uint31_t ntohs(uint32_t net32bitvalue); 頭文件: #include<netinet/in.h> 把net32bitvalue有網絡字節序轉換為主機字節序。 */ if(send(new_fd,"Hello client,I am 192.168.229.125!\n",50,0)==-1) //192.168.229.125為子進程IP,可更改 perror("send failed"); pid=fork(); //父進程建立套接字的連接之后,創建子進程用於通信 if(pid<0) perror("fork error\n"); if(!pid)//創建新的子進程,用於發送數據 { // close(sock_fd);//子進程不需要監聽,所以子進程關閉監 while(fgets(sendbuf,sizeof(sendbuf),stdin)!=NULL) { writen(new_fd,sendbuf,strlen(sendbuf)); memset(sendbuf,0,sizeof(sendbuf)); //清空,以免和下一次混淆 // exit(EXIT_SUCCESS); } exit(EXIT_SUCCESS); } //close(new_fd);//父進程不需要連接,所以關閉連接套接字 else { while(1) { memset(recvbuf,0,sizeof(recvbuf)); ret=readline(new_fd,recvbuf,1024); if(ret<0) perror("read from client error"); else if(ret==0) { printf("peer closed\n"); break; } fputs(recvbuf,stdout); } exit(EXIT_SUCCESS); } //while(waitpid(-1,NULL,WNOHANG)>0);//等待子進程結束,進行新的連接 //} return 0; }
客戶端代碼
#include<stdio.h> #include<stdlib.h> #include<errno.h> #include<string.h> #include<sys/types.h> #include<sys/socket.h> #include<sys/un.h> #include<sys/wait.h> //*進程用的頭文件*/ #include<netinet/in.h> #include<arpa/inet.h> #define MAXBYTEMUN 1024 ssize_t readn(int fd, void *buf, size_t count) { ssize_t nleft=count; ssize_t nread; char *charbuf=(char*) buf; while(nleft>0) { nread=read(fd,charbuf,nleft); if(nread<0) { if(errno==EINTR) continue; return -1; } else if(nread==0) return count-nleft; charbuf +=nread; nleft=count-nread; } return count; } ssize_t writen(int fd, const void *buf, size_t count) { ssize_t nleft=count; ssize_t nwrite; char *charbuf=(char*) buf; while(nleft>0) { nwrite=write(fd,charbuf,nleft); if(nwrite<0) { if(errno==EINTR) continue; return -1; } else if(nwrite==0) return count-nleft; charbuf +=nwrite; nleft=count-nwrite; } return count; } ssize_t recv_peek(int sockfd, void *buf, size_t len) { int ret; while(1) { ret=recv(sockfd,buf,len,MSG_PEEK); if(ret==-1&& errno==EINTR) continue; return ret; } } ssize_t readline(int sockfd, void *buf, size_t len) { ssize_t nleft=len,nread; int ret; char* bufchar=buf; while(1) { ret=recv_peek(sockfd,bufchar,len); if(ret<0||ret==0) return ret; nread=ret; int i; for(i=0;i<nread;i++) { if(bufchar[i]=='\n') { ret=readn(sockfd,bufchar,i+1); if(ret!=i+1) exit(EXIT_FAILURE); return ret; } } if(nread>nleft) exit(EXIT_FAILURE); nleft-=nread; ret=readn(sockfd,bufchar,nread); if(ret!=nread) exit(EXIT_FAILURE); bufchar+=nread; } return -1; } int main(int argc,char *argv[]) { int sock_fd,numbytes; char buf[MAXBYTEMUN]; struct hostent *he; struct sockaddr_in client_addr;//客戶機的地址信息 ssize_t n,ret; char recvbuf[1024]={'0'},sendbuf[1024]={'0'}; if(argc!=2) { fprintf(stderr,"usage: client IPAddress\n"); //執行客戶端程序時,輸入客戶端程序名稱和其IP地址 exit(1); } /*創建套接字*/ sock_fd=socket(AF_INET,SOCK_STREAM,0);//采用IPv4協議 if(sock_fd==-1) { perror("creat socket failed"); exit(1); } /*服務器地址參數*/ client_addr.sin_family=AF_INET; client_addr.sin_port=htons(3490); client_addr.sin_addr.s_addr=inet_addr(argv[1]); bzero(&client_addr.sin_zero,sizeof(struct sockaddr_in));//bzero位清零函數,將sin_zero清零,sin_zero為填充字段,必須全部為零 /*連接到服務器*/ if(connect(sock_fd,(struct sockaddr*)&client_addr,sizeof(struct sockaddr))==-1) { perror("connect failed"); exit(1); } if((numbytes=recv(sock_fd,buf,MAXBYTEMUN,0))==-1) { perror("receive failed"); exit(1); } buf[numbytes]='\0';//在字符串末尾加上\0,否則字符串無法輸出 printf("Received: %s\n",buf); pid_t pid; pid=fork(); if(!pid)//創建新的子進程,用於接收數據 { while(1) { memset(recvbuf,0,sizeof(recvbuf)); ret=readline(sock_fd,recvbuf,1024); if(ret<0) perror("read from server error"); else if(ret==0) { printf("peer closed\n"); break; } fputs(recvbuf,stdout); } close(sock_fd); } else //f=父進程用於發送數據 { //close(sock_fd);//父進程不需要連接,所以關閉連接套接字 while(fgets(sendbuf,sizeof(sendbuf),stdin)!=NULL) { writen(sock_fd,sendbuf,strlen(sendbuf)); memset(sendbuf,0,sizeof(sendbuf)); //清空,以免和下一次混淆 } close(sock_fd); } return 0; }