Linux TCP服务器 - 用C ++监听多个端口

时间:2013-08-27 12:28:43

标签: c++ linux unix tcp port

我遇到了tcp服务器的问题。我想听多个端口来响应客户端。它应该是一种基于事件的。每个端口都表示另一种响应。我读了很多关于epoll,poll,select或multithreading的内容。我尝试使用Unix网络编程等书中的大量例子。但可能我需要一些触发关键字。我怎么能正常开始?

希望我的问题很容易理解。 感谢每个答案!

让我知道我的想法...... 我开始想到这个:

如果我有一台带有大量服务器的“服务器管理器”,我可以按如下方式进行操作吗?

CreateSockets(SERVERLIST); CheckSockets(SocketList,master_set);

在服务器管理器中: 1)for循环创建所有服务器套接字(函数:socket / setsockopt / ioctl / bind / listen)

void CreateSockets(map<int,ServerType> ServerList)
{
    fd_set        master_set;
    map<int,ServerType>::iterator it;
    map<int,int> SocketList;
    for (it= ServerList.begin();it!= ServerList.end();it++)
    {
        listen_sd = socket(AF_INET, SOCK_STREAM, 0);
        if (listen_sd < 0)
        {
            perror("socket() failed");
            exit(-1);
        }
        rc = setsockopt(listen_sd, SOL_SOCKET,  SO_REUSEADDR,
                (char *)&on, sizeof(on));
        if (rc < 0)
        {
            perror("setsockopt() failed");
            close(listen_sd);
            exit(-1);
        }
        rc = ioctl(listen_sd, FIONBIO, (char *)&on);
        if (rc < 0)
        {
            perror("ioctl() failed");
            close(listen_sd);
            exit(-1);
        }
        memset(&addr, 0, sizeof(addr));
        addr.sin_family      = AF_INET;
        addr.sin_addr.s_addr = htonl(INADDR_ANY);
        addr.sin_port        = ((*it).second->Port);
        rc = bind(listen_sd,(struct sockaddr *)&addr, sizeof(addr));
        if (rc < 0)
        {
            perror("bind() failed");
            close(listen_sd);
            exit(-1);
        }
        rc = listen(listen_sd, 32);
        if (rc < 0)
        {
            perror("listen() failed");
            close(listen_sd);
            exit(-1);
        }
        SocketList.insert(make_pair(((*it).second->Port),listen_sd));
        FD_ZERO(&master_set);
        max_sd = listen_sd;
        FD_SET(listen_sd, &master_set);
    }
}

下一步:  2)有些如何等待服务器管理器中的某些事件(使用套接字描述符列表选择)

void CheckSockets(map<int,int> SocketList, fd_set master_set)
 {
    fd_set working_set;
    do
    {
        memcpy(&working_set, &master_set, sizeof(master_set));
        ready_descriptors = select(max_sd + 1, &working_set, NULL, NULL, NULL);

        if (ready_descriptors >0)
        {
            desc_ready = rc;
            for (i=0; i <= max_sd  &&  desc_ready > 0; ++i)
            {
                if (FD_ISSET(i, &working_set))
                {
                    desc_ready -= 1;
                    if (i == listen_sd)
                    {
                        do
                        {
                            new_sd = accept(listen_sd, NULL, NULL);
                            if (new_sd < 0)
                            {
                                //error
                            }
                            FD_SET(new_sd, &master_set);
                            if (new_sd > max_sd)
                                max_sd = new_sd;
                        } while (new_sd != -1);
                    }
                    else
                    {
                        do
                        {
                            //Go into server and recv and send ( Input Parameter = i)
                            CheckServer(i);
                        } while (TRUE);
                    }
                }
            }
        }
        else {endserver=true;}
    }while(endserver=true;)

}

3)进入服务器并处理问题(recv / send)????

    void CheckServer( int sd)
 {
    rc = recv(sd, buffer, sizeof(buffer), 0);

    //some stuff in between

    rc = send(i, buffer, len, 0); 
 }

这可行吗?

从IBM非阻塞IO源代码中使用和更改了某些部分。


谢谢你的帮助。我能够完成一些事情,但有一件事仍然无法运作。

到目前为止我做了什么:

1)个人服务器的consrtuctor包括套接字操作。 2)我能够返回套接字ID并将其保存在服务器管理器中。 3)管理器有一个for循环,其中包含select命令以检查套接字上的任何事件。 4)如果发生了什么事情,所有受影响的套接字将按顺序重复。

我的问题是:

如果我在从服务器请求数据时始终连接和断开连接,它可以正常工作。 当我的客户端以一种连接将被保持的方式配置时,一切都在阻塞,因为我的代码正在等待断开连接。

以下是每个部分的代码段:

1)

       Server::Server() 
    {

    listen_sd = socket(AF_INET, SOCK_STREAM, 0);
    ret = setsockopt(listen_sd, SOL_SOCKET,  SO_REUSEADDR,(char *)&on, sizeof(on));
    ret = ioctl(listen_sd, FIONBIO, (char *)&on);

    memset(&addr, 0, sizeof(addr));
    addr.sin_family      = AF_INET;
    addr.sin_addr.s_addr = htonl(INADDR_ANY);
    addr.sin_port        = htons(Server_Port);
    ret = bind(listen_sd,(struct sockaddr *)&addr, sizeof(addr));
    ret = listen(listen_sd, 32);
    Socket = listen_sd;
}

2)

Socket= new_Server->GetSocket();
SocketList.insert(make_pair(Socket,new_Server->ServerID));

3)

 while (TRUE)
    {
        FD_ZERO(&working_set);
        for (i=0;i < max_conn;i++) 
{
            if (SocketArray[i] >= 0) {FD_SET(SocketArray[i], &working_set);}
        }

        ret = select(max_sd+1, &working_set, NULL, NULL, NULL);

    desc_ready= ret;
        for (i=0; i <= max_sd  &&  desc_ready > 0; ++i)
        {

            if (FD_ISSET(i, &working_set)) //jeder Peer der was hat
            {
                desc_ready -= 1;
//delete all loops to get the correct object 
                        (Server).second->DoEvent(i);

            }
        }
    }

4)

new_sd = accept(new_sd, NULL, NULL);
    if (new_sd < 0)
    {
        if (errno != EWOULDBLOCK)
        {
            perror("  accept() failed");
        }
    }
do
{


    rc = recv(new_sd, buffer, sizeof(buffer), 0);
//edit datastream and create response
            rc = send(new_sd, buffer, len, 0);
        if (rc < 0)
        {
            perror("  send() failed");
            close_conn = TRUE;
            break;
        }

    }while (TRUE);

我只是删除了错误处理dor listen / bind等,只是为了缩短这里的代码......原来它就在那里。

1 个答案:

答案 0 :(得分:2)

这里的步骤大致如下:您可以让多个TCP服务器(也就是服务器套接字)监听每个端口。接下来,您可以为每个服务器套接字使用select()和传递文件描述符。如果你在其中任何一个上获得连接,那么select将返回一个读事件并标记具有连接的服务器套接字的fd。你需要在该服务器fd上调用accept()。

你不能让一个TCP套接字监听多个端口。