从HttpClient和HttpURLConnection使用到其实现原理

435 查看

序言

从我们的最开始使用的HttpClient到HttpURLConnection,当然现在如果还在说你在项目中自己封装使用它们两个,有点多余了,无论是Volley还是OkHttp都是要比其好很多的,写起来方便,效率高,你就是要造轮子,这个就无法阻挡你了,拥有一个造轮子的心,还要记得,性非议也善假于物也。今天回顾下,基本使用,然后从一个post到网络端接收的一个处理过程,进行一个分析,让我们对于其底层的实现有所了解。而现在,HttpClient已经被弃用,所以这里也不再讲,来讲下HttpURLConnection。

HttpURLConnection

HttpURLConnection相比于HttpClient,其API简单,体积小,而且其压缩和缓存机制可以有效的减少网络访问的流量,在提升速度和省电方面都很有优势,

public void sendRequest(String url) throws IOException{
        InputStream is = null;
        try{
            URL newUrl = new URL(url);
            HttpURLConnection conn = (HttpURLConnection)newUrl.openConnection();
            conn.setReadTimeout(10000);
            conn.setConnectTimeout(10000);
            conn.setRequestMethod("POST");
            conn.setDoOutput(true);
            conn.setDoOutput(true);
            conn.setRequestProperty("Connection", "Keep-Alive");
            String data = "username=jensea";
            OutputStream out = conn.getOutputStream();// 获得一个输出流,向服务器写数据
            out.write(data.getBytes());
            out.flush();
            out.close();
            conn.connect();
            is = conn.getInputStream();
            String result = convertStreamToString(is);
        }catch (IOException e){
            
        }finally {
            is.close();
        }
    }

    private String convertStreamToString(InputStream is) throws IOException{
        BufferedReader reader = new BufferedReader(new InputStreamReader(is));
        StringBuilder sb = new StringBuilder();
        String line = null;
        try{
            while((line=reader.readLine())!=null){
                sb.append(line+"/n");
            }
        }catch (IOException e){
            e.printStackTrace();
        }
        return sb.toString();
    }

从客户端到服务端看网络请求实现

对于客户端和服务端,其实现无非是通过对Socket进行读写,然后对其进行解析,根据解析内容作出响应,然后客户端从自己的socket缓冲区中读取信息,而我们所使用的HttpClient,HttpURLConnection则是对其一个封装,将其中的流读写的细节进行了一个隐藏,而我们的Volley,OkHttp则是在其基础上,结合线程池进行了一个更高效的封装。

服务端

public class SimpleHttpServer extends Thread {
    public static final int HTTP_PORT = 8000;
    ServerSocket mSocket = null;

    public SimpleHttpServer(){
        try{
            mSocket = new ServerSocket(HTTP_PORT);
        }catch (IOException e){
            e.printStackTrace();
        }
        if(mSocket == null){
            throw new RuntimeException("服务器Socket初始化失败");
        }
    }

    @Override
    public void run() {
        try {
            while(true){
                System.out.print("等待连接中");
                new DeliverThread(mSocket.accept()).start();
            }
        }catch (IOException e){
            e.printStackTrace();
        }
    }
}
  • 绑定制定端口套接字

  • 等待连接请求,创建处理线程

对于具体的请求内容处理交给了Deliver线程

public class DeliverThread extends Thread {

    Socket mClientSocket;
    BufferedReader mInputStream;
    PrintStream mOutputStream;
    //请求方法
    String httpMethod;
    String subPath;
    String boundary;
    //请求参数
    Map<String, String> mParams = new HashMap<String, String>();

    Map<String, String> mHeaders = new HashMap<String, String>();

    boolean isParseHeader = false;

    public DeliverThread(Socket socket) {
        mClientSocket = socket;
    }

    @Override
    public void run() {
        try {
            mInputStream = new BufferedReader(new
                    InputStreamReader(mClientSocket.getInputStream()));
            mOutputStream = new PrintStream(mClientSocket.getOutputStream());
            parseRequest();
            handleResponse();
        } catch (IOException e) {

        }
    }
//处理请求,读出每一行,确定是头部,还是请求体,然后交给不同的函数处理
    private void parseRequest() {
        String line;
        try {
            int lineNum = 0;
            while ((line = mInputStream.readLine()) != null){
                if(lineNum==0){
                    parseRequestLine(line);
                }
                if(isEnd(line)){
                    break;
                }
                if(lineNum!=0&&!isParseHeader){
                    parseHeader(line);
                }
                //头部被处理完,处理请求参数
                if(isParseHeader){
                    parseRequestparams(line);
                }
                lineNum++;
            }
        } catch (IOException e) {
            e.printStackTrace();
        }

    }
    
    //处理请求行,请求方法+空格+HTTP版本号
    private void parseRequestLine(String lineOne) {
        String[] tmpStrings = lineOne.split(" ");
        httpMethod = tmpStrings[0];
        subPath = tmpStrings[1];
    }
    //处理头部
    private void parseHeader(String headLine) {
        if (headLine.equals("")) {
            isParseHeader = true;
            return;
        } else if (headLine.contains("boundary")) {
            boundary = parseSecondField(headLine);
        }else{
            parseHeaderParam(headLine);
        }
    }

    private String parseSecondField(String line) {
        String[] headerArray = line.split(";");
        parseHeaderParam(headerArray[0]);
        if(headerArray.length>1){
            return headerArray[1].split("=")[1];
        }
        return "";

    }
    //处理头部每一个参数,然后将其添加到mHeaders中
    private void parseHeaderParam(String headerLine) {
        String[] keyvalue = headerLine.split(":");
        mHeaders.put(keyvalue[0].trim(), keyvalue[1].trim());
    }

    //处理请求参数
    private void parseRequestparams(String paramLine) throws IOException{
        if(paramLine.equals("--"+boundary)){
            String ContentDisposition = mInputStream.readLine();
            String paraName = parseSecondField(ContentDisposition);
            mInputStream.readLine();
            String paramvalue = mInputStream.readLine();
            mParams.put(paraName,paramvalue);
        }
    }
    
    //根据传递的参数进行相应的处理
    private void handleResponse(){
        mOutputStream.println("HTTP/1.1 200 OK");
        mOutputStream.println("Content_Type:application/json");
        mOutputStream.println();
        mOutputStream.println("{\"stcode\":\"success\"}");

    ]
}

上述代码设计到一些字符串的处理,相对也比较简单,对于其中请求参数的处理,你可能会感觉有点难懂。首先要明白对于其要处理的数据中,请求的数据格式是怎么样的。

--DFDFJD3243dfddfdfdffdf
content-Disposition:form-data; name="username"

Mr.Simple

通过其格式,再去回顾一下上面的处理方式就不难看懂了。

客户端

public class HttpPost {
    public String url;
    private Map<String,String> mParamsMap = new HashMap<String,String>();
    Socket mSocket;

    public HttpPost(String url){
        this.url = url;
    }

    public void addParam(String key,String value){
        mParamsMap.put(key,value);
    }

    public void execute(){
        try{
            mSocket = new Socket(this.url,SimpleHttpServer.HTTP_PORT);
            PrintStream outputStream = new PrintStream(mSocket.getInputStream());
            BufferedReader inputStream = new BufferedReader(new InputStreamReader(mSocket
                    .getInputStream()));
            final String boundary = "my_boundary_123";
            writeHeader(boundary,outputStream);
            writeParams(boundary,outputStream);
            waitResponse(inputStream);
        }catch (UnknownHostException e){
            e.printStackTrace();
        }catch (IOException e){
            e.printStackTrace();
        }finally {
            mSocket.close();
        }
    }

    private void writeHeader(String boundary,PrintStream outputStream){
        outputStream.println("POST /api/login/ HTTP/1.1");
        outputStream.println("content-length:123");
        outputStream.println("Host:"+this.url);
        outputStream.println("boundary="+boundary);
        outputStream.println("Uset-Agent:android");
        outputStream.println();
    }

    private void writeParams(String boundary,PrintStream outputStream){
        Iterator<String> paramsKeySet = mParamsMap.keySet().iterator();
        while(paramsKeySet.hasNext()){
            String paramName = paramsKeySet.next();
            outputStream.println("--"+boundary);
            outputStream.println("Content-Disposition:form-data; name="+paramName);
            outputStream.println();
            outputStream.println(mParamsMap.get(paramName));
        }
        outputStream.println("--"+boundary+"--");
    }

    private void waitResponse(BufferedReader inputStream) throws IOException{
        String responseLine = inputStream.readLine();
        while(responseLine==null||!responseLine.contains("HTTP")){
            responseLine = inputStream.readLine();
        }
        while((responseLine=inputStream.readLine())!=null){
            System.out.println(responseLine);
        }
    }
}

代码量不大,创建socket,然后向socket中,写入头部,请求参数,等待服务器的回应,但是这个过程中,socket又干了些什么呢?socket又是个什么呢?这个可以参考我前面对于网络和Linux下进程通信的文章。