Tomcat源码分析(一):HTTP协议解析

前几天面试阿里,面试官问我如何解析HTTP协议,我大概说了一下的思路,他最后得出的结论是我对HTTP协议不了解,让我很受打击。回来看《深入剖析Tomcat》,研究一下Tomcat是如何解析HTTP协议的

1. 环境说明

  • 《深入剖析Tomcat》是基于tomcat-4.1.12进行分析,这个版本在2002年发布,可以说是老古董了。不过就学习而言还是很好的工具.
  • Http协议的解析在连接器(connector) 中进行,连接器是一个独立的模块,可以被插入到容器中,tomcat-4.1.12里提供了默认连接器,但已被标注为过时。

2. 源码分析

2.1 连接器

默认连接器在org.apache.catalina.connector.http包下,它实现了Connector接口和Runnable接口。分析的入口在run方法中

2.1.1 run()

public void run() {
    while (!stopped) {
        //从ServerSocket中接受下一个进入的连接
        Socket socket = null;
        try {
            serverSocket.accept()");
            socket = serverSocket.accept();
            if (connectionTimeout > 0)
                socket.setSoTimeout(connectionTimeout);
            socket.setTcpNoDelay(tcpNoDelay);//这个有点意思,关闭TCP延迟确认
        } catch (AccessControlException ace) {
            log("socket accept security exception", ace);
            continue;
        } catch (IOException e) {
            try {
                // 如果重新打开失败,退出
                synchronized (threadSync) {
                    if (started && !stopped)
                        log("accept error: ", e);
                    if (!stopped) {
                        serverSocket.close();
                        serverSocket = open();
                    }
                }
            } catch (IOException ioe) {
                log("socket reopen, io problem: ", ioe);
                break;
            } catch (KeyStoreException kse) {
                log("socket reopen, keystore problem: ", kse);
                break;
            } catch (NoSuchAlgorithmException nsae) {
                log("socket reopen, keystore algorithm problem: ", nsae);
                break;
            } catch (CertificateException ce) {
                log("socket reopen, certificate problem: ", ce);
                break;
            } catch (UnrecoverableKeyException uke) {
                log("socket reopen, unrecoverable key: ", uke);
                break;
            } catch (KeyManagementException kme) {
                log("socket reopen, key management problem: ", kme);
                break;
            }
            continue;
        }
        // 把socket给适当的处理器
        HttpProcessor processor = createProcessor();//2.1.2
        if (processor == null) {
            try {
                log(sm.getString("httpConnector.noProcessor"));
                socket.close();
            } catch (IOException e) {
                ;
            }
            continue;
        }
        processor.assign(socket);//2.2.3
        // The processor will recycle itself when it finishes
    }
    synchronized (threadSync) {
        threadSync.notifyAll();
    }
}

2.1.2 createProcessor()

具体的处理将在HttpProcessor中进行,一个连接器会创建多个处理器,连接器的数量通过maxProcessorsminProcessors进行控制。今天的重点在http协议的解析,创建HttpProcessor的一些细节就不说了

private HttpProcessor createProcessor() {
    synchronized (processors) {
        if (processors.size() > 0) {
            return ((HttpProcessor) processors.pop());
        }
        if ((maxProcessors > 0) && (curProcessors < maxProcessors)) {
            return (newProcessor());
        } else {
            if (maxProcessors < 0) {
                return (newProcessor());
            } else {
                return (null);
            }
        }
    }
}

2.2 处理器

HttpProcessor在独立的线程中对请求进行处理,连接器将请求分配给处理器(调用处理器的assign()方法),处理器处理完成后将进行回收重复利用

2.2.1 run()

HttpProcessor同样实现了Runnable接口,在后台一直运行(被设置为守护线程),等待处理请求

public void run() {
    while (!stopped) {
        // 等待下一个socket
        Socket socket = await();//2.2.2
        if (socket == null)
            continue;
        // 处理请求
        try {
            process(socket);//2.2.4
        } catch (Throwable t) {
            log("process.invoke", t);
        }
        // 完成此次请求
        connector.recycle(this);
    }
    synchronized (threadSync) {
        threadSync.notifyAll();
    }
}

2.2.2 await()

await()监视available变量,如果没有新的请求,就进入阻塞状态,同时run()方法也会被阻塞

private synchronized Socket await() {
    // Wait for the Connector to provide a new Socket
    while (!available) {
        try {
            wait();
        } catch (InterruptedException e) {
        }
    }
    // Notify the Connector that we have received this Socket
    Socket socket = this.socket;
    available = false;
    notifyAll();
    if ((debug >= 1) && (socket != null))
        log("  The incoming request has been awaited");
    return (socket);
}

2.2.3 assign(Socket socket)

连接器调用assign方法分配请求,它会唤醒阻塞的线程.这实际上是一个生产者-,消费者模型,通过available变量,将请求从连接器传递到处理器。但这个实现并不优雅,并且效率也不高

synchronized void assign(Socket socket) {
    // Wait for the Processor to get the previous Socket
    while (available) {
        try {
            wait();
        } catch (InterruptedException e) {
        }
    }
    // Store the newly available Socket and notify our thread
    this.socket = socket;
    available = true;
    notifyAll();
    if ((debug >= 1) && (socket != null))
        log(" An incoming request is being assigned");
}

2.2.4 process(Socket socket)

process(Socket socket)方法对请求进行处理,此处省略了很多

private void process(Socket socket) {
    boolean ok = true;
    boolean finishResponse = true;
    SocketInputStream input = null;
    OutputStream output = null;
    // 构造和初始化需要的对象
    try {
        input = new SocketInputStream(socket.getInputStream(),
                                      connector.getBufferSize());
    } catch (Exception e) {
        log("process.create", e);
        ok = false;
    }
    keepAlive = true;
    while (!stopped && ok && keepAlive) {
        finishResponse = true;
        try {
            //此处的request,response是循环利用的
            request.setStream(input);
            request.setResponse(response);
            output = socket.getOutputStream();
            response.setStream(output);
            response.setRequest(request);
            ((HttpServletResponse) response.getResponse()).setHeader
                ("Server", SERVER_INFO);
        } catch (Exception e) {
            //...
        }
        // 解析请求
        try {
            if (ok) {
                parseConnection(socket);//2.2.5
                parseRequest(input, output);//2.2.6
                if (!request.getRequest().getProtocol().startsWith("HTTP/0"))
                    parseHeaders(input);//2.2.8
                if (http11) {
                    // 若在请求头中发现"EXpect:100-continue",则设置sendAck为true
                    //ackRequest方法检查sendAck的值和是否允许分块,如果为true向客户端发送HTTP/1.1 100 Continue\r\n\r\n
                    ackRequest(output);
                    // If the protocol is HTTP/1.1, chunking is allowed.
                    if (connector.isChunkingAllowed())
                        response.setAllowChunking(true);
                }
            }
        } catch (EOFException e) {
            //很可能client或server中的一方断开连接
            ok = false;
            finishResponse = false;
        } catch (ServletException e) {
            //...
        } catch (InterruptedIOException e) {
            //...
        } catch (Exception e) {
            //...
        }
        try {
            ((HttpServletResponse) response).setHeader("Date", FastHttpDateFormat.getCurrentDate());
            if (ok) {
                connector.getContainer().invoke(request, response);//如果处理正常调用容器的invoke方法
            }
        } catch (ServletException e) {
            //...
        } catch (InterruptedIOException e) {
            //...
        } catch (Throwable e) {
            //...
        }
        // 完成处理请求
        if (finishResponse) {
            //省略...
            //主要是调用response.finishResponse();
        }
        //必须检查Connection是否被设置为close或者在HTTP/1.0下
        if ( "close".equals(response.getHeader("Connection")) ) {
            keepAlive = false;
        }
        // 如果keepAlive为true并且解析没有发生错误,则继续while循环
        status = Constants.PROCESSOR_IDLE;
        // 回收request和response对象
        request.recycle();
        response.recycle();
    }
    try {
        shutdownInput(input);
        socket.close();
    } catch (IOException e) {
        //...
    } catch (Throwable e) {
        //...
    }
    socket = null;
}

2.2.5 parseConnection(Socket socket)

解析连接信息,获取Internet地址,检查是否使用代理

private void parseConnection(Socket socket)
    throws IOException, ServletException {
    if (debug >= 2)
        log("  parseConnection: address=" + socket.getInetAddress() +
            ", port=" + connector.getPort());
    ((HttpRequestImpl) request).setInet(socket.getInetAddress());
    if (proxyPort != 0)
        request.setServerPort(proxyPort);
    else
        request.setServerPort(serverPort);
    request.setSocket(socket);
}

2.2.6 parseRequest(SocketInputStream input, OutputStream output)

requestLine是一个HttpRequest实例,其中包含3个char[],分别对应method,uri,protocol.调用SocketInputStreamreadRequestLine()方法填充请求行,再获得对应的请求方法,URI,协议版本,(查询参数,session ID)

private void parseRequest(SocketInputStream input, OutputStream output)
    throws IOException, ServletException {
    // 解析请求行
    input.readRequestLine(requestLine);//2.2.7
    status = Constants.PROCESSOR_ACTIVE;
    String method = new String(requestLine.method, 0, requestLine.methodEnd);//获得请求方法
    String uri = null;
    String protocol = new String(requestLine.protocol, 0, requestLine.protocolEnd);//获得协议版本信息
    if (protocol.length() == 0)
        protocol = "HTTP/0.9";
    // 如果是HTTP/1.1需要在解析请求后保持连接
    if ( protocol.equals("HTTP/1.1") ) {
        http11 = true;
        sendAck = false;
    } else {
        http11 = false;
        sendAck = false;
        // 对于HTTP/1.0, 默认不保持连接,除非指定Connection:Keep-Alive
        keepAlive = false;
    }
    // 验证请求行
    if (method.length() < 1) {
        throw new ServletException(sm.getString("httpProcessor.parseRequest.method"));
    } else if (requestLine.uriEnd < 1) {
        throw new ServletException(sm.getString("httpProcessor.parseRequest.uri"));
    }
    // 解析URI上的查询参数
    int question = requestLine.indexOf("?");
    if (question >= 0) {
        request.setQueryString(new String(requestLine.uri, question + 1,requestLine.uriEnd - question - 1));//设置查询参数
        if (debug >= 1)
            log(" Query string is " +
                ((HttpServletRequest) request.getRequest())
                .getQueryString());
        uri = new String(requestLine.uri, 0, question);//获得URI
    } else {
        request.setQueryString(null);
        uri = new String(requestLine.uri, 0, requestLine.uriEnd);
    }
    // Checking for an absolute URI (with the HTTP protocol)
    //检验绝对URI路径和HTTP协议
    if (!uri.startsWith("/")) {
        int pos = uri.indexOf("://");
        // 解析协议和主机名
        if (pos != -1) {
            pos = uri.indexOf('/', pos + 3);
            if (pos == -1) {
                uri = "";
            } else {
                uri = uri.substring(pos);
            }
        }
    }
    // 从请求URI解析session ID
    int semicolon = uri.indexOf(match);//match=";jsessionid="0
    if (semicolon >= 0) {
        String rest = uri.substring(semicolon + match.length());
        int semicolon2 = rest.indexOf(';');
        if (semicolon2 >= 0) {
            request.setRequestedSessionId(rest.substring(0, semicolon2));//设置session ID
            rest = rest.substring(semicolon2);
        } else {
            request.setRequestedSessionId(rest);
            rest = "";
        }
        request.setRequestedSessionURL(true);
        uri = uri.substring(0, semicolon) + rest;
        if (debug >= 1)
            log(" Requested URL session id is " +
                ((HttpServletRequest) request.getRequest())
                .getRequestedSessionId());
    } else {
        request.setRequestedSessionId(null);
        request.setRequestedSessionURL(false);
    }
    //修正RUI(使用字符串操作)
    String normalizedUri = normalize(uri);
    if (debug >= 1) log("Normalized: '" + uri + "' to '" + normalizedUri + "'");
    // 设置请求属性
    ((HttpRequest) request).setMethod(method);//设置请求方法
    request.setProtocol(protocol);//设置协议版本
    if (normalizedUri != null) {
        ((HttpRequest) request).setRequestURI(normalizedUri);
    } else {
        ((HttpRequest) request).setRequestURI(uri);
    }
    request.setSecure(connector.getSecure());
    request.setScheme(connector.getScheme());
    if (normalizedUri == null) {
        log(" Invalid request URI: '" + uri + "'");
        throw new ServletException("Invalid URI: " + uri + "'");
    }
    if (debug >= 1)
        log(" Request is '" + method + "' for '" + uri +
            "' with protocol '" + protocol + "'");
}

2.2.7 readRequestLine(HttpRequestLine requestLine)

readRequestLine方法会分别填充请求行,URI,协议版本

public void readRequestLine(HttpRequestLine requestLine)
    throws IOException {
    // 检查是否已回收
    if (requestLine.methodEnd != 0)
        requestLine.recycle();
    // 检查空白行
    int chr = 0;
    do { // 跳过 CR(\r) 或 LF(\n)
        try {
            chr = read();
        } catch (IOException e) {
            chr = -1;
        }
    } while ((chr == CR) || (chr == LF));
    if (chr == -1)
        throw new EOFException (sm.getString("requestStream.readline.error"));
    pos--;
    // 读取方法名
    int maxRead = requestLine.method.length;//这里的char[]数组的长度为8
    int readStart = pos;
    int readCount = 0;
    boolean space = false;
    //读取到空格说明方法名已解析完成
    while (!space) {
        // 如果char[]已满,将容量翻倍
        if (readCount >= maxRead) {
            if ((2 * maxRead) <= HttpRequestLine.MAX_METHOD_SIZE) {
                char[] newBuffer = new char[2 * maxRead];
                System.arraycopy(requestLine.method, 0, newBuffer, 0,
                                 maxRead);
                requestLine.method = newBuffer;
                maxRead = requestLine.method.length;
            } else {
                throw new IOException
                    (sm.getString("requestStream.readline.toolong"));
            }
        }
        // 检查是否读取到末尾
        if (pos >= count) {
            int val = read();
            if (val == -1) {
                throw new IOException
                    (sm.getString("requestStream.readline.error"));
            }
            pos = 0;
            readStart = 0;
        }
        // 检查是否读取到空格
        if (buf[pos] == SP) {
            space = true;
        }
        //填充char[] method
        requestLine.method[readCount] = (char) buf[pos];
        readCount++;
        pos++;
    }
    requestLine.methodEnd = readCount - 1;//设置请求方法结束位置
    // 解析URI
    maxRead = requestLine.uri.length;
    readStart = pos;
    readCount = 0;
    space = false;
    boolean eol = false;
    while (!space) {
        if (readCount >= maxRead) {
            if ((2 * maxRead) <= HttpRequestLine.MAX_URI_SIZE) {
                char[] newBuffer = new char[2 * maxRead];
                System.arraycopy(requestLine.uri, 0, newBuffer, 0,
                                 maxRead);
                requestLine.uri = newBuffer;
                maxRead = requestLine.uri.length;
            } else {
                throw new IOException(sm.getString("requestStream.readline.toolong"));
            }
        }
        // 检查是否读取到末尾
        if (pos >= count) {
            int val = read();
            if (val == -1)
                throw new IOException(sm.getString("requestStream.readline.error"));
            pos = 0;
            readStart = 0;
        }
        // 检查是否读取到空格
        if (buf[pos] == SP) {
            space = true;
        } else if ((buf[pos] == CR) || (buf[pos] == LF)) {
            // HTTP/0.9 风格的请求
            eol = true;
            space = true;
        }
        //填充 char[] uri
        requestLine.uri[readCount] = (char) buf[pos];
        readCount++;
        pos++;
    }
    requestLine.uriEnd = readCount - 1;//设置uri结束位置
    // 解析协议
    maxRead = requestLine.protocol.length;
    readStart = pos;
    readCount = 0;
    //是否结束
    while (!eol) {
        if (readCount >= maxRead) {
            if ((2 * maxRead) <= HttpRequestLine.MAX_PROTOCOL_SIZE) {
                char[] newBuffer = new char[2 * maxRead];
                System.arraycopy(requestLine.protocol, 0, newBuffer, 0,
                                 maxRead);
                requestLine.protocol = newBuffer;
                maxRead = requestLine.protocol.length;
            } else {
                throw new IOException(sm.getString("requestStream.readline.toolong"));
            }
        }
         // 检查是否读取到末尾
        if (pos >= count) {
            int val = read();
            if (val == -1)
                throw new IOException(sm.getString("requestStream.readline.error"));
            pos = 0;
            readStart = 0;
        }
        //是否结束
        if (buf[pos] == CR) {
            // 跳过\r
        } else if (buf[pos] == LF) {
            eol = true;
        } else {
            //填充char[] protocol
            requestLine.protocol[readCount] = (char) buf[pos];
            readCount++;
        }
        pos++;
    }
    requestLine.protocolEnd = readCount;//设置协议版本结束位置
}

2.2.8 parseHeaders(SocketInputStream input)

一个HttpHeader包含一个name数组和value数组,通过SocketInputStream中的readHeader方法填充HttpHeader对象,整个过程和readRequestLine类似.
通过HttpHeader对象,设置request对象对应的属性

private void parseHeaders(SocketInputStream input)
    throws IOException, ServletException {
    while (true) {
        HttpHeader header = request.allocateHeader();//分配一个HttpHeader对象,从对象池中
        // 解析请求头
        input.readHeader(header);
        if (header.nameEnd == 0) {
            if (header.valueEnd == 0) {
                return;
            } else {
                throw new ServletException
                    (sm.getString("httpProcessor.parseHeaders.colon"));
            }
        }
        String value = new String(header.value, 0, header.valueEnd);//获得value值
        if (debug >= 1)
            log(" Header " + new String(header.name, 0, header.nameEnd)+ " = " + value);
        // 设置对应的请求头
        if (header.equals(DefaultHeaders.AUTHORIZATION_NAME)) {//authorization头
            request.setAuthorization(value);
        } else if (header.equals(DefaultHeaders.ACCEPT_LANGUAGE_NAME)) {//accept-language头
            parseAcceptLanguage(value);
        } else if (header.equals(DefaultHeaders.COOKIE_NAME)) {//cookie头
            Cookie cookies[] = RequestUtil.parseCookieHeader(value);//将value解析成Cookie数组
            for (int i = 0; i < cookies.length; i++) {
                if (cookies[i].getName().equals
                    (Globals.SESSION_COOKIE_NAME)) {//判断cookie名是否为JSESSIONID
                    if (!request.isRequestedSessionIdFromCookie()) {
                        // 只接受第一个session ID
                        request.setRequestedSessionId(cookies[i].getValue());//设置session ID
                        request.setRequestedSessionCookie(true);
                        request.setRequestedSessionURL(false);
                        if (debug >= 1)
                            log(" Requested cookie session id is " +
                                ((HttpServletRequest) request.getRequest())
                                .getRequestedSessionId());
                    }
                }
                if (debug >= 1)
                    log(" Adding cookie " + cookies[i].getName() + "=" +
                        cookies[i].getValue());
                request.addCookie(cookies[i]);//添加cookie到request对象
            }
        } else if (header.equals(DefaultHeaders.CONTENT_LENGTH_NAME)) {//content-length头
            int n = -1;
            try {
                n = Integer.parseInt(value);
            } catch (Exception e) {
                throw new ServletException
                    (sm.getString("httpProcessor.parseHeaders.contentLength"));
            }
            request.setContentLength(n);
        } else if (header.equals(DefaultHeaders.CONTENT_TYPE_NAME)) {//content-type头
            request.setContentType(value);
        } else if (header.equals(DefaultHeaders.HOST_NAME)) {//host头
            int n = value.indexOf(':');
            if (n < 0) {
                if (connector.getScheme().equals("http")) {
                    request.setServerPort(80);//设置http协议端口
                } else if (connector.getScheme().equals("https")) {
                    request.setServerPort(443);//设置https协议端口
                }
                if (proxyName != null)
                    request.setServerName(proxyName);
                else
                    request.setServerName(value);
            } else {
                if (proxyName != null)
                    request.setServerName(proxyName);
                else
                    request.setServerName(value.substring(0, n).trim());
                if (proxyPort != 0)
                    request.setServerPort(proxyPort);
                else {
                    int port = 80;
                    try {
                        port =Integer.parseInt(value.substring(n+1).trim());
                    } catch (Exception e) {
                        throw new ServletException
                            (sm.getString("httpProcessor.parseHeaders.portNumber"));
                    }
                    request.setServerPort(port);
                }
            }
        } else if (header.equals(DefaultHeaders.CONNECTION_NAME)) {//connection头
            if (header.valueEquals(DefaultHeaders.CONNECTION_CLOSE_VALUE)) {//close值
                keepAlive = false;
                response.setHeader("Connection", "close");
            }
        } else if (header.equals(DefaultHeaders.EXPECT_NAME)) {//expect头
            if (header.valueEquals(DefaultHeaders.EXPECT_100_VALUE))//100-continue值
                sendAck = true;
            else
                throw new ServletException
                    (sm.getString("httpProcessor.parseHeaders.unknownExpectation"));
        } else if (header.equals(DefaultHeaders.TRANSFER_ENCODING_NAME)) {//transfer-encoding头
            //request.setTransferEncoding(header);
        }
        request.nextHeader();//读取下一个请求头
    }
}

2.3 Request对象

在调用getParameter,getParameterMap,getParameterNames,getParameterValues时会先调用parseParameters方法解析请求参数

2.3.1 parseParameters()

parseParameters方法将解析结果放入ParameterMap对象中;ParameterMap基础自HashMap,添加了锁定属性,当被锁定时不允许修改

protected void parseParameters() {
    if (parsed)//如果已解析直接返回
        return;
    ParameterMap results = parameters;//初始化ParameterMap对象
    if (results == null)
        results = new ParameterMap();
    results.setLocked(false);//解除锁定
    String encoding = getCharacterEncoding();//获得编码信息
    if (encoding == null) encoding = "ISO-8859-1";//默认编码
    // 解析查询字符串中的参数
    String queryString = getQueryString();
    try {
        RequestUtil.parseParameters(results, queryString, encoding);//解析查询字符串
    } catch (UnsupportedEncodingException e) {
        ;
    }
    // 从正文中的参数
    String contentType = getContentType();
    if (contentType == null)
        contentType = "";
    int semicolon = contentType.indexOf(';');
    if (semicolon >= 0) {
        contentType = contentType.substring(0, semicolon).trim();
    } else {
        contentType = contentType.trim();
    }
    if ("POST".equals(getMethod()) && (getContentLength() > 0)&& (this.stream == null)
        && "application/x-www-form-urlencoded".equals(contentType)) {
        //判断条件:POST方法,content-length>0,有ServletInputStream,content-type=application/x-www-form-urlencoded
        try {
            int max = getContentLength();
            int len = 0;
            byte buf[] = new byte[getContentLength()];
            ServletInputStream is = getInputStream();
            while (len < max) {//读取数据
                int next = is.read(buf, len, max - len);
                if (next < 0 ) {
                    break;
                }
                len += next;
            }
            is.close();
            if (len < max) {
                //FIX ME,当实际接收长度小于content-length声明的长度时
                //上面的代码中检查next=-1可以预防出现死循环
                //但是这个bug必须在mod_jk模块中
                //记录额外的信息用于debug mod_jk
                StringBuffer msg = new StringBuffer();
                msg.append("HttpRequestBase.parseParameters content length mismatch\n");
                msg.append("  URL: ");
                msg.append(getRequestURL());
                msg.append(" Content Length: ");
                msg.append(max);
                msg.append(" Read: ");
                msg.append(len);
                msg.append("\n  Bytes Read: ");
                if ( len > 0 ) {
                    msg.append(new String(buf,0,len));
                }
                log(msg.toString());
                throw new RuntimeException
                    (sm.getString("httpRequestBase.contentLengthMismatch"));
            }
            RequestUtil.parseParameters(results, buf, encoding);//解析参数
        } catch (UnsupportedEncodingException ue) {
            ;
        } catch (IOException e) {
            throw new RuntimeException
                    (sm.getString("httpRequestBase.contentReadFail") + 
                     e.getMessage());
        }
    }
    results.setLocked(true);
    parsed = true;
    parameters = results;
}

至此整个HTTP协议的解析流程就完成了

3. 总结

  • 连接器负责接收请求,处理器负责解析请求,每个处理器拥有自己的Request和Response对象,这两个对象可以重复使用
  • 处理器处理流程
    1. 解析连接信息:设置Internet地址和代理信息
    2. 解析请求行:请求方法,URI,协议版本,查询参数(如果有),keep-alive属性,session ID(如果禁用了cookie),标准化URI地址
    3. 解析请求头:将请求头设置到对应的属性中,其中有几个重要的属性,cookie,content-length和content-type(在处理正文时会用到),conection(主要检查是否close值)
    4. 解析参数:先解析URI中的查询参数,再解析正文中的参数
    5. 调用容器的invoke方法

PS:
看完Tomcat如何解析才发现自己对HTTP协议不了解,只考虑到了HTTP协议的格式,没有考虑到不同版本的区别,特殊请求头的处理,不同请求方法的处理,cookie的解析,session的处理。
随着HTTP协议的发展,解析的难度越来越大,要求也越来越高(高效+正确);以上的代码在Tomcat4中已经废弃,换了更高效的连接器.等有时间去看一下Tomcat8的源码

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,580评论 18 139
  • Http协议详解 标签(空格分隔): Linux 声明:本片文章非原创,内容来源于博客园作者MIN飞翔的HTTP协...
    Sivin阅读 5,201评论 3 82
  • 1.OkHttp源码解析(一):OKHttp初阶2 OkHttp源码解析(二):OkHttp连接的"前戏"——HT...
    隔壁老李头阅读 20,799评论 24 176
  • 一、概念(载录于:http://www.cnblogs.com/EricaMIN1987_IT/p/3837436...
    yuantao123434阅读 8,326评论 6 152
  • 外面的世界很精彩,外面的世界很无奈,外面的世界很孤独…… 当我坐着长达五六个小时的大巴,跨省,只为了参加一次考试。...
    二月木子禽阅读 287评论 0 0