为什么没有收到另一个http请求?

时间:2013-06-09 18:42:20

标签: java http nio

我正在使用Java NIO编写一个简单的HTTP服务器,但很早就陷入了困境。我有以下代码:

Selector accept = Selector.open();
            ServerSocketChannel ssc = ServerSocketChannel.open();
            ssc.configureBlocking(false);

            InetAddress lh = InetAddress.getByName("127.0.0.1");
            InetSocketAddress isa = new InetSocketAddress(lh, port);
            ssc.socket().bind(isa);
            SelectionKey acceptKey = ssc.register(accept,
                    SelectionKey.OP_ACCEPT);

            while (accept.select() > 0) {
                Set<SelectionKey> readyKeys = accept.selectedKeys();
                Iterator<SelectionKey> i = readyKeys.iterator();

                while (i.hasNext()) {
                    SelectionKey sk = i.next();
                    if (sk.isAcceptable()) {
                        System.out.println("Is acceptable");
                        ssc = (ServerSocketChannel) sk.channel();
                        SocketChannel sc = ssc.accept();
                        sc.configureBlocking(false);
                        sc.register(accept, SelectionKey.OP_READ);
                        System.out.println("Registered new SocketChannel");
                    }
                    if (sk.isReadable()) {
                        SocketChannel sc = (SocketChannel) sk.channel();
                        ByteBuffer buffer = ByteBuffer.allocate(20000);
                        buffer.clear();
                        int bytesRead = sc.read(buffer);
                        buffer.flip();

                        while (buffer.hasRemaining()) {
                            System.out.print((char) buffer.get());
                        }

                        buffer.clear();
                    }
                    i.remove();
                }
            }

现在,如果我在浏览器中打开两个选项卡并导航到localhost:8080,这将是应用程序的输出:

Is acceptable
Registered new SocketChannel
Is acceptable
Registered new SocketChannel
GET / HTTP/1.1
Host: localhost:8080
Connection: keep-alive
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
User-Agent: Mozilla/5.0 (X11; Linux i686) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/27.0.1453.110 Safari/537.36
Accept-Encoding: gzip,deflate,sdch
Accept-Language: en-US,en;q=0.8

Is acceptable
Registered new SocketChannel

现在我有两个问题: 1)为什么我在开始时会得到额外的接受事件? 2)为什么没有收到第二个http请求?接受连接,其SocketChannel正在选择器中注册。但是没有收到请求正文。我知道生成了许多“空”读事件但没有一个带来请求体。

2 个答案:

答案 0 :(得分:1)

  1. 你有三个接受事件。
  2. 可能是因为你没有回复第一个。
  3. '空'读取事件可能会关闭,您忽略了。
  4. 尝试修复所有这些并重新测试。当你不是时,你不能指望同伴表现自己。

答案 1 :(得分:1)

  1. 为什么我在开始时会收到额外的接受事件?

    • 仅当您从浏览器发起请求时才会发生这种情况。
    • 如果使用简单的socket / http客户端创建连接,则只会收到一个接受请求。 (虽然在读取返回-1时需要修改读取代码以关闭套接字。)
  2. 为什么没有收到第二个http请求?

  3. 在IE

    中检查此行为
    • 修改代码以在连接关闭时执行读取时处理异常(在catch块sc.close()和sk.cancel()中)
    • 从IE
    • 发起请求
    • 两次接受+一次阅读
    • 在IE中取消请求
    • 等待一段时间,然后在同一个标​​签页中再次发起请求
    • 只收到一次阅读,没有接受请求
    • 现在再次转到相同标签并取消请求
    • 现在再次从同一个标签
    • 发起请求
    • 相同的两个接受+一个读
    • 上面让我相信#2点可能会为备份目的创建第二个连接