C中的多客户端服务器实现

时间:2013-01-25 09:30:53

标签: c client-server

我正在尝试使用C实现多客户端服务器。我已经为服务器编写代码,客户端是一个将不断向服务器发送一些数据包的软件。服务器将读取数据包并进行处理。我一次尝试5连接到服务器。但是我写的服务器代码存在一些问题,因为我无法将5个客户端连接到服务器。确切的问题是,当客户端尝试与服务器连接时,它会获得连接,但是当我关闭客户端软件并尝试重新启动时,它没有获得连接。以下是我的服务器端代码。任何人都可以帮我解决这个问题。

#include <ctype.h>
#include <sys/time.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/utsname.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/select.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h>
#include <unistd.h>
#include <sys/file.h>


int sock;                       /* The socket file descriptor for our "listening"socket */
int connectlist[15];            /* Array of connected sockets so we know who we are talking to */
fd_set socks;                   /* Socket file descriptors we want to wake up for, using select() */
int highsock = 1;               /* Highest #'d file descriptor, needed for select() */
struct sockaddr_in client_address[5];
unsigned int clientLength = sizeof(client_address) ;
#define PORTNO      (int)49153
int port;                       /* The port number after conversion from ascport */
struct sockaddr_in server_address; /* bind info structure */
int reuse_addr = 1;  
struct timeval timeout;         /* Timeout for select */
int readsocks;                  /* Number of sockets ready for reading */
int err = 0 ;

#define BACKLOG         (int)10

void deal_with_data(int listnum /* Current item in connectlist for for loops */)
{
    //Here I am trying to read packet from client s/w and process it
}

void setnonblocking(int sock)
{
    int opts;
    opts = fcntl(sock,F_GETFL);
    if (opts < 0) 
    {
        printf("fcntl(F_GETFL)_error");
        exit(0);
    }
    opts = (opts | O_NONBLOCK);
    if (fcntl(sock,F_SETFL,opts) < 0) 
    {
        printf("fcntl(F_SETFL)_error");
        exit(0);
    }
    return;
}

void build_select_list() 
{
    int listnum;         /* Current item in connectlist for for loops */
    FD_ZERO(&socks);
    FD_SET(sock,&socks);
    for (listnum = 0; listnum < 5; listnum++) 
    {
        if (connectlist[listnum] != 0) 
        {
            FD_SET(connectlist[listnum],&socks);
            if (connectlist[listnum] > highsock)
                highsock = connectlist[listnum];
        }
    }
}

void handle_new_connection() {
    int listnum;         /* Current item in connectlist for for loops */
    int connection; /* Socket file descriptor for incoming connections */
    connection = accept(sock, (struct sockaddr *)&client_address[highsock], &clientLength);
    if (connection < 0) 
    {
        printf("accept_error");
        exit(0);
    }
    setnonblocking(connection);
    for (listnum = 0; (listnum < 5) && (connection != -1); listnum ++)
        if (connectlist[listnum] == 0) 
        {
            printf("\nConnection accepted:   FD=%d; Slot=%d\n",
                connection,listnum);
            printf("Connection accepted from %s\n",inet_ntoa(client_address[highsock].sin_addr));
            connectlist[listnum] = connection;
            connection = -1;

        }
    if (connection != -1) 
    {
        printf("\nNo room left for new client.\n");
        write(connection,"Sorry, this server is too busy.Try again later!\r\n",80);
        close(connection);
    }
    printf("return from handle_new_connection\n");
}

void read_socks(void)
{
    int listnum;         /* Current item in connectlist for for loops */
    if (FD_ISSET(sock,&socks))
        handle_new_connection();
    for (listnum = 0; listnum < 5; listnum++) 
    {
        if (FD_ISSET(connectlist[listnum],&socks))
        {
            //printf("read_socks2\n");
            deal_with_data(listnum);
        }
    }
}

int main (/*int argc, char *argv[]*/)
{
    sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if (sock < 0) 
    {
        printf("socket_error");
        exit(EXIT_FAILURE);
    }
    setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &reuse_addr,sizeof(reuse_addr));
    setnonblocking(sock);

    memset((char *) &server_address, 0, sizeof(server_address));
    server_address.sin_family = AF_INET;
    server_address.sin_addr.s_addr = htonl(INADDR_ANY);
    server_address.sin_port = htons(port);

    if (bind(sock, (struct sockaddr *) &server_address,sizeof(server_address)) < 0 ) 
    {
        printf("bind_error");
        close(sock);
        exit(EXIT_FAILURE);
    }

    if((err = listen(sock,10)) == -1)
    {
        printf("listen_error");
    }

    highsock = sock;
    memset((char *) &connectlist, 0, sizeof(connectlist));

    while (1) 
    { /* Main server loop - forever */
        build_select_list();
        timeout.tv_sec = 2;
        timeout.tv_usec = 0;

        readsocks = select(highsock+2, &socks, (fd_set *) 0,(fd_set *) 0, &timeout);
        if (readsocks < 0) 
        {
            printf("select_error");
            exit(EXIT_FAILURE);
        }
        if (readsocks == 0) 
        {
            printf(".");
            fflush(stdout);
        } 
        else
        {   
            read_socks();
        }
    } /* while(1) */
} /* main */

2 个答案:

答案 0 :(得分:2)

您的问题是,当客户端关闭套接字时,您的主循环不会退出。这意味着它无法接受新连接。我会使用fork()来处理来自套接字的数据,使用main()函数来接受连接和fork()进程。此外,您需要有一些代码可以杀死fork() ed(即检查客户端是否在fork() ed进程中断开连接)进程(因为它不会自行关闭并且需要记忆)。

编辑:

好的,我在你的程序中找不到recv()的电话。根据{{​​3}},如果客户端强行断开连接,则return 0如果客户端“正常”断开连接,return -1并将errno设置为ECONNRESET。为了使用fork,我会(在main()函数中)将while循环包装在:

int childpid = fork();
if(childpid == -1) {
       printf("Could not fork process");
       exit(EXIT_FAILURE);
}
else if(childpid == 0) { /* in child process*/
       while(/* check if the socket has been closed */) {
               /* While loop stuff */
       }
       /* free up memory */
       exit(EXIT_SUCCESS);
}

你的主要功能应该在一个循环中,等待新的连接。

N.B。我没有测试过这些代码,所以它可能无法正常工作。但是,如果您阅读了fork(3)recv(3)的手册页,它应该有用。

答案 1 :(得分:0)

最好不要像Apache那样使用现成的产品并根据需要进行自定义吗?