如何将简单的客户端服务器TCP程序转换为非阻塞程序

时间:2015-03-19 17:59:26

标签: c select tcp nonblocking

您好我正在阅读Beej指南中使用select()的非阻止来电,但我仍然对如何将我的简单客户端 - 服务器代码更改为非阻塞。谁能告诉我在服务器代码中需要做哪些更改以及客户端代码?

这是服务器代码:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h> 
#include <sys/socket.h>
#include <netinet/in.h>

void error(const char *msg)
{
    perror(msg);
    exit(1);
}

int main(int argc, char *argv[])
{
     int sockfd, newsockfd, portno;
     socklen_t clilen;
     char buffer[256];
     struct sockaddr_in serv_addr, cli_addr;
     int n;
     if (argc < 2) {
         fprintf(stderr,"ERROR, no port provided\n");
         exit(1);
     }
     sockfd = socket(AF_INET, SOCK_STREAM, 0);
     if (sockfd < 0) 
        error("ERROR opening socket");
     bzero((char *) &serv_addr, sizeof(serv_addr));
     portno = atoi(argv[1]);
     serv_addr.sin_family = AF_INET;
     serv_addr.sin_addr.s_addr = INADDR_ANY;
     serv_addr.sin_port = htons(portno);
     if (bind(sockfd, (struct sockaddr *) &serv_addr,
              sizeof(serv_addr)) < 0) 
              error("ERROR on binding");
     listen(sockfd,5);
     clilen = sizeof(cli_addr);
     newsockfd = accept(sockfd, 
                 (struct sockaddr *) &cli_addr, 
                 &clilen);
     if (newsockfd < 0) 
          error("ERROR on accept");
     bzero(buffer,256);
     n = read(newsockfd,buffer,255);
     if (n < 0) error("ERROR reading from socket");
     printf("Here is the message: %s\n",buffer);
     n = write(newsockfd,"I got your message",18);
     if (n < 0) error("ERROR writing to socket");
     close(newsockfd);
     close(sockfd);
     return 0; 
}

以下是客户端代码:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h> 

void error(const char *msg)
{
    perror(msg);
    exit(0);
}

int main(int argc, char *argv[])
{
    int sockfd, portno, n;
    struct sockaddr_in serv_addr;
    struct hostent *server;

    char buffer[256];
    if (argc < 3) {
       fprintf(stderr,"usage %s hostname port\n", argv[0]);
       exit(0);
    }
    portno = atoi(argv[2]);
    sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd < 0) 
        error("ERROR opening socket");
    server = gethostbyname(argv[1]);
    if (server == NULL) {
        fprintf(stderr,"ERROR, no such host\n");
        exit(0);
    }
    bzero((char *) &serv_addr, sizeof(serv_addr));
    serv_addr.sin_family = AF_INET;
    bcopy((char *)server->h_addr, 
         (char *)&serv_addr.sin_addr.s_addr,
         server->h_length);
    serv_addr.sin_port = htons(portno);
    if (connect(sockfd,(struct sockaddr *) &serv_addr,sizeof(serv_addr)) < 0) 
        error("ERROR connecting");
    printf("Please enter the message: ");
    bzero(buffer,256);
    fgets(buffer,255,stdin);
    n = write(sockfd,buffer,strlen(buffer));
    if (n < 0) 
         error("ERROR writing to socket");
    bzero(buffer,256);
    n = read(sockfd,buffer,255);
    if (n < 0) 
         error("ERROR reading from socket");
    printf("%s\n",buffer);
    close(sockfd);
    return 0;
}

1 个答案:

答案 0 :(得分:0)

select(2)使非阻塞的方式是等待I / O成为可能而不会阻塞(例如,当新数据可用时)。使用select()时,通常不必将受监控的描述符置于非阻塞模式(例如SOCK_NONBLOCK),因此在某种意义上select()具体是关于避免必须使用非阻塞I / O.

select()用于一次等待多个描述符上的事件。这里的事件可以使read(2)或(取决于您如何使用select()write(2)到描述符而不阻塞。

例如,您可以使用select()同时等待服务器中已连接客户端的新客户端连接和数据(假设您将其扩展为处理多个客户端)。为此,您可以使用select()来监控sockfd和从accept(2)返回的任何描述符。如果没有select(),则必须使用某种形式的非阻塞I / O(或单独的线程)来避免陷入困境。 accept()直到新客户端连接,这会阻止您在此期间看到来自其他客户端的数据。实施起来既麻烦又效率低于在单一地点睡觉的效率。

select()本身不做任何I / O.只有在没有阻塞的情况下I / O成为可能时它才会通知您。你传递了一组描述符,它会告诉你每当I / O成为可能时。 (还会告诉你哪些描述符是可能的。)

除了等待套接字上的数据外,您还可以使用select()等待,例如用户同时输入stdin。有许多不同类型的描述符可以select()编辑。