TCP中 recv和sendf函數


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;
}

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM