草庐IT

c - 监听两个不同端口的 TCP 服务器

coder 2023-09-18 原文

我有一个 TCP 服务器正在监听两个不同的端口。我创建了两个不同的套接字,一个在端口 8888 上,一个在端口 6634 上。我监听这些端口,然后我在 FD_SET 中添加两个套接字并将它们传递给 select() 函数...... 当套接字准备好读取时,我检查 FD_ISSET 以查看我在哪个端口上有消息要读取。

无论如何,当我连接到端口 8888 时,构思是成功的,我可以向服务器发送并接收...当我 ctrl+c 客户端时,选择功能再次返回 1,现在我的 accept() 失败了.. . 当我在端口 6634 上做同样的事情时,一切正常......代码在 select() 处停止并等待套接字准备好读取!

谁能告诉我为什么会这样?

看附件中我的代码

    int main()
    {
        SOCKET          conn_request_skt;   /* socket where connections are accepted */
        char            buf[RBUFLEN], buf1[RBUFLEN];        /* reception buffer */
        uint16_t        lport_n, lport_h, lport_n1, lport_h1;   /* port where the server listens (net/host byte ord resp.) */
        int         bklog = 2;      /* listen backlog */
        SOCKET          s,s1;           
        int         result, n;
        socklen_t addrlen;
        struct sockaddr_in  saddr, caddr;       /* server and client address structures */ 
        int optval,childpid,i; /* flag value for setsockopt */
        int connectcnt; /* number of connection requests */
        fd_set readfds;

        /* Initialize socket API if needed */
        SockStartup();

        /* input server port number */
        lport_h=6634;
        lport_n = htons(lport_h);
        lport_h1=8888;
        lport_n1 = htons(lport_h1);

        /* create the socket */
        printf("Creating first socket\n");
        s = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
        if (s == INVALID_SOCKET)
            err_fatal("socket() failed");
        printf("done, socket number %u\n",s);

        /* bind the socket to any local IP address */
        saddr.sin_family      = AF_INET;
        saddr.sin_port        = lport_n;
        saddr.sin_addr.s_addr = INADDR_ANY;
        showAddr("Binding to address first socket", &saddr);
        result = bind(s, (struct sockaddr *) &saddr, sizeof(saddr));
        if (result == -1)
            err_fatal("bind() failed");
        printf("done.\n");

        printf("Creating second socket\n");
        s1 = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
        if (s1 == INVALID_SOCKET)
            err_fatal("socket() failed");
        printf("done, socket number %u\n",s1);

        /* bind the socket to any local IP address */

        saddr.sin_port=lport_n1;

        showAddr("Binding to address second socket", &saddr);
        result = bind(s1, (struct sockaddr *) &saddr, sizeof(saddr));
        if (result == -1)
            err_fatal("bind() failed");
        printf("done.\n");


        /* listen */
        printf ("Listening at socket %d with backlog = %d \n",s,bklog);
        result = listen(s, bklog);
        if (result == -1)
            err_fatal("listen() failed");
        printf("done.\n");

        printf ("Listening at socket %d with backlog = %d \n",s1,bklog);
        result = listen(s1, bklog);
        if (result == -1)
            err_fatal("listen() failed");
        printf("done.\n");
for (;;)
    {


        FD_ZERO(&readfds);          /* initialize the fd set */
        FD_SET(s, &readfds);
        FD_SET(s1, &readfds); /* add socket fd */
        printf("here \n");

        printf("result bifore select is %d \n", result);
        result=select(s1+1, &readfds, 0, 0, 0);

        printf("result after select is %d \n", result);

        if(result<0)
            {
                err_fatal("select() failed");

            }
        if(result>0)
        {

            if(FD_ISSET(s,&readfds))
            {

                conn_request_skt=s;
                /* accept next connection */
                addrlen = sizeof(struct sockaddr_in);
                s = accept(conn_request_skt, (struct sockaddr *) &caddr, &addrlen);
                if (s == INVALID_SOCKET)
                err_fatal("accept() failed");
                showAddr("Accepted connection from", &caddr);
                printf("new socket: %u\n",s);
                /* serve the client on socket s */
                for (;;)
                {
                    n=recv(s, buf, RBUFLEN-1, 0);
                    if (n < 0)
                    {
                        printf("Read error\n");
                        closesocket(s);
                        printf("Socket %d closed\n", s);
                        break;
                    }
                    else if (n==0)
                    {
                        printf("Connection closed by party on socket %d\n",s);
                        //closesocket(s);
                        break;
                    }
                    else
                    {
                        printf("Received line from socket %03d :\n", s);
                        buf[n]=0;
                        printf("[%s]\n",buf);
                        if(writen(s, buf, n) != n)
                        printf("Write error while replying\n");
                        else
                        printf("Reply sent\n");
                    }   
                }
            }


            if(FD_ISSET(s1,&readfds))
            {
                conn_request_skt=s1;
                /* accept next connection */
                addrlen = sizeof(struct sockaddr_in);
                printf("bifore accept! \n");
                s1 = accept(conn_request_skt, (struct sockaddr *) &caddr, &addrlen);
                if (s1 == INVALID_SOCKET)
                err_fatal("accept() failed");
                showAddr("Accepted connection from", &caddr);
                printf("new socket: %u\n",s1);
                /* serve the client on socket s */
                for (;;)
                {
                    n=recv(s1, buf, RBUFLEN-1, 0);
                    if (n < 0)
                    {
                        printf("Read error\n");
                        closesocket(s1);
                        printf("Socket %d closed\n", s1);
                        break;
                    }
                    else if (n==0)
                    {
                        printf("Connection closed by party on socket %d\n",s1);
                        //closesocket(s);
                        break;
                    }
                    else
                    {
                        printf("Received line from socket %03d :\n", s1);
                        buf[n]=0;
                        printf("[%s]\n",buf);
                        if(writen(s1, buf, n) != n)
                        printf("Write error while replying\n");
                        else
                        printf("Reply sent\n");
                    }   
                }
            }
        }
    }
}

最佳答案

第一个监听器套接字是通过以下方式创建的:

s = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);

然后数据套接字被接受:

 conn_request_skt=s;
 s = accept(conn_request_skt, (struct sockaddr *) &caddr, &addrlen);

看到了吗?下一个循环,当您要选择监听器套接字时,s 不再保留该套接字,而是(关闭的)数据套接字。

解决方案是为监听器套接字和数据套接字使用不同的变量(conn_request_skt 只是混淆了这个问题)。

关于c - 监听两个不同端口的 TCP 服务器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32741710/

有关c - 监听两个不同端口的 TCP 服务器的更多相关文章

  1. ruby - 使用 ruby​​ 和 savon 的 SOAP 服务 - 2

    我正在尝试使用ruby​​和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我

  2. ruby - 具有身份验证的私有(private) Ruby Gem 服务器 - 2

    我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..

  3. ruby-on-rails - 如何在 ruby​​ 中使用两个参数异步运行 exe? - 2

    exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby​​中使用两个参数异步运行exe吗?我已经尝试过ruby​​命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何ruby​​gems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除

  4. ruby-on-rails - 启动 Rails 服务器时 ImageMagick 的警告 - 2

    最近,当我启动我的Rails服务器时,我收到了一长串警告。虽然它不影响我的应用程序,但我想知道如何解决这些警告。我的估计是imagemagick以某种方式被调用了两次?当我在警告前后检查我的git日志时。我想知道如何解决这个问题。-bcrypt-ruby(3.1.2)-better_errors(1.0.1)+bcrypt(3.1.7)+bcrypt-ruby(3.1.5)-bcrypt(>=3.1.3)+better_errors(1.1.0)bcrypt和imagemagick有关系吗?/Users/rbchris/.rbenv/versions/2.0.0-p247/lib/ru

  5. ruby-on-rails - s3_direct_upload 在生产服务器中不工作 - 2

    在Rails4.0.2中,我使用s3_direct_upload和aws-sdkgems直接为s3存储桶上传文件。在开发环境中它工作正常,但在生产环境中它会抛出如下错误,ActionView::Template::Error(noimplicitconversionofnilintoString)在View中,create_cv_url,:id=>"s3_uploader",:key=>"cv_uploads/{unique_id}/${filename}",:key_starts_with=>"cv_uploads/",:callback_param=>"cv[direct_uplo

  6. ruby - 用 Ruby 编写一个简单的网络服务器 - 2

    我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b

  7. ruby - 这两个 Ruby 类初始化定义有什么区别? - 2

    我正在阅读一本关于Ruby的书,作者在编写类初始化定义时使用的形式与他在本书前几节中使用的形式略有不同。它看起来像这样:classTicketattr_accessor:venue,:datedefinitialize(venue,date)self.venue=venueself.date=dateendend在本书的前几节中,它的定义如下:classTicketattr_accessor:venue,:datedefinitialize(venue,date)@venue=venue@date=dateendend在第一个示例中使用setter方法与在第二个示例中使用实例变量之间是

  8. ruby-on-rails - 在 Rails 中调试生产服务器 - 2

    您如何在Rails中的实时服务器上进行有效调试,无论是在测试版/生产服务器上?我试过直接在服务器上修改文件,然后重启应用,但是修改好像没有生效,或者需要很长时间(缓存?)我也试过在本地做“脚本/服务器生产”,但是那很慢另一种选择是编码和部署,但效率很低。有人对他们如何有效地做到这一点有任何见解吗? 最佳答案 我会回答你的问题,即使我不同意这种热修补服务器代码的方式:)首先,你真的确定你已经重启了服务器吗?您可以通过跟踪日志文件来检查它。您更改的代码显示的View可能会被缓存。缓存页面位于tmp/cache文件夹下。您可以尝试手动删除

  9. java - 为什么 ruby​​ modulo 与 java/other lang 不同? - 2

    我基本上来自Java背景并且努力理解Ruby中的模运算。(5%3)(-5%3)(5%-3)(-5%-3)Java中的上述操作产生,2个-22个-2但在Ruby中,相同的表达式会产生21个-1-2.Ruby在逻辑上有多擅长这个?模块操作在Ruby中是如何实现的?如果将同一个操作定义为一个web服务,两个服务如何匹配逻辑。 最佳答案 在Java中,模运算的结果与被除数的符号相同。在Ruby中,它与除数的符号相同。remainder()在Ruby中与被除数的符号相同。您可能还想引用modulooperation.

  10. ruby - 具有两个参数的 block - 2

    我从用户Hirolau那里找到了这段代码:defsum_to_n?(a,n)a.combination(2).find{|x,y|x+y==n}enda=[1,2,3,4,5]sum_to_n?(a,9)#=>[4,5]sum_to_n?(a,11)#=>nil我如何知道何时可以将两个参数发送到预定义方法(如find)?我不清楚,因为有时它不起作用。这是重新定义的东西吗? 最佳答案 如果您查看Enumerable#find的文档,您会发现它只接受一个block参数。您可以将它发送两次的原因是因为Ruby可以方便地让您根据它的“并行赋

随机推荐