HTTP分解一

http分解一

    首先,我不在这个地方来说这个http是个啥东西,不会了看完整定义去,记住,我这个地方是通过代码实现来告诉你http具体是个啥玩意。其次,我们只考虑这个http协议层传输,先不要考虑tcp/ip和物理层的数据传输。这个传输是基于socket套接字建立起来的。

好!现在我们先来看一下这个socket是个啥玩意,因为这个http是建立在socket套接字上的。看完下面的实现代码我将让你明白http的作用是啥。

package com.devil.java.practice.secoket;

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.Socket;
import java.net.UnknownHostException;

/**
 * socket客户端,用于发起一次链接请求
 * 
 * @author devil
 *
 */
public class SocketClient {

    public static void main(String[] args) {
        // 指定访问的服务端的主机地址
        String host = "127.0.0.1";
        // 指定链接的端口号
        int port = 8181;
        // 我们开始链接服务端
        try {
            Socket socket = new Socket(host, port);
            DataInputStream din = new DataInputStream(socket.getInputStream());
            DataOutputStream dout = new DataOutputStream(socket.getOutputStream());
            String sayClient = "hello server!";
            String sayServer = "";
            dout.writeUTF(sayClient);
            dout.flush();
            sayServer = din.readUTF();
            System.out.println("Server says: " + sayServer);
            dout.close();
            socket.close();
        } catch (UnknownHostException e) {
            System.out.println(e.getMessage());
        } catch (IOException e) {
            System.out.println(e.getMessage());
        }

    }

}
package com.devil.java.practice.secoket;

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * socket服务端
 * 
 * @author devil
 *
 */
public class SocketService {

    public static void main(String[] args) {

        int port = 8181;
        ServerSocket server = null;
        ExecutorService executors = Executors.newFixedThreadPool(5);
        try {
            // 我们启动服务端监听127.0.0.1的8181端口
            server = new ServerSocket(port);
            while (true) {
                // 我们只链接一次,如果需要可以做一个循环一直监听
                Socket socket = server.accept();
                executors.execute(new Server(socket));
            }

        } catch (UnknownHostException e) {
            System.err.println(e.getMessage());
        } catch (IOException e) {
            System.err.println(e.getMessage());

        }
    }

    public static class Server implements Runnable {

        private Socket socket;

        public Server(Socket socket) {
            this.socket = socket;
        }

        public void run() {
            try {
                DataInputStream din = new DataInputStream(socket.getInputStream());
                DataOutputStream dout = new DataOutputStream(socket.getOutputStream());
                String sayClient = "";
                String sayServer = "hello client!";
                sayClient = din.readUTF();
                System.out.println("Client says: " + sayClient);
                dout.writeUTF(sayServer);
                dout.flush();
                socket.close();
            } catch (IOException e) {
                System.out.println(e.getMessage());
            }

        }

    }
}
Server says: hello client!
Client says: hello server!
 通上面的代码可以简单的实现通信,当然这个通信之间没有任何的协议,就是单纯的字符串的传递。那么我们可以考虑一下,如果我们需要传递的是一个文件那么应该咋做?接下来我们先讲一下这个http协议的格式。

http协议的格式

一、 请求的格式
「请求的方法·空格·URL·空格·协议版本·回车符·换行符」(请求行)
「头部字段名·冒号·值·回车符·换行符」(请求的头部)
「······」
「头部字段名·冒号·值·回车符·换行符」
「回车符·换行符」
「请求的数据」(请求的数据)

GET /devil?index=1 HTTP/1.1
Host: 127.0.0.1:8181
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10.12; rv:51.0) Gecko/20100101 Firefox/51.0
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: zh-CN,zh;q=0.8,en-US;q=0.5,en;q=0.3
Accept-Encoding: gzip, deflate
Connection: keep-alive
Upgrade-Insecure-Requests: 1
Cache-Control: max-age=0

二、 响应的格式
「版本协议·空格·状态码·空格·状态码的描述·回车符·换行符」(请求行)
「头部字段名·冒号·值·回车符·换行符」(请求的头部)
「······」
「头部字段名·冒号·值·回车符·换行符」
「回车符·换行符」
「请求的数据」(请求的数据)

HTTP/1.1 200 OK
Date: Fri Mar 03 11:00:43 CST 2017
Content-Type: text/html; charset=utf-8
Content-Length: 12

Hello World!

我们改一下服务端的代码通过浏览器请求看是不是这个结果

package com.devil.java.practice.secoket;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintStream;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.UnknownHostException;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * socket服务端
 * 
 * @author devil
 *
 */
public class SocketService {

    public static void main(String[] args) {

        int port = 8181;
        ServerSocket server = null;
        ExecutorService executors = Executors.newFixedThreadPool(5);
        try {
            // 我们启动服务端监听127.0.0.1的8181端口
            server = new ServerSocket(port);
            while (true) {
                // 我们只链接一次,如果需要可以做一个循环一直监听
                Socket socket = server.accept();
                executors.execute(new Server(socket));
            }

        } catch (UnknownHostException e) {
            System.err.println(e.getMessage());
        } catch (IOException e) {
            System.err.println(e.getMessage());

        }
    }

    public static class Server implements Runnable {

        private Socket socket;

        public Server(Socket socket) {
            this.socket = socket;
        }

        public void run() {
            try {
                BufferedReader read = new BufferedReader(new InputStreamReader(socket.getInputStream()));
                PrintStream print = new PrintStream(socket.getOutputStream());
                String sayClient = "";
                System.out.println("-----------------------请求头中的值-------------------------");
                Map<String, String> hear = new HashMap<String, String>();
                sayClient = read.readLine();
                while (sayClient.length() != 0) {
                    // 显示请求头的信息
                    System.out.println(sayClient);
                    // 去掉包含的空格
                    sayClient = sayClient.trim();
                    // 通过冒号取出键值对
                    String[] k = sayClient.split(":");
                    if (k.length == 2) {
                        hear.put(k[0], k[1]);
                    }
                    // 读取下一个请求头的值
                    sayClient = read.readLine();
                }
                System.out.println(hear);
                System.out.println("-----------------------BODY 中的值-------------------------");
                // 获取body中的字节长度
                String length = hear.get("Content-Length");
                // 如果存在键值对,那么有值我们读取body中的数据
                if (length != null) {
                    length = length.trim();
                    char[] b = new char[Integer.valueOf(length)];
                    read.read(b);
                    System.out.println(new String(b));
                }
                System.out.println("-----------------------READ END-------------------------");
                // 返回给客户端的数据
                String result = "Hello World!";
                // 拼接响应头
                print.println("HTTP/1.1 200 OK");
                print.println("Date: " + new Date());
                print.println("Content-Type: text/html; charset=utf-8");
                print.println("Content-Length: " + result.getBytes().length);
                // 空行
                print.println();
                // 响应的数据
                print.println(result);
                print.flush();
                System.out.println("-----------------------PRINT END-------------------------");
                socket.close();
            } catch (IOException e) {
                System.out.println(e.getMessage());
            }

        }

    }
}

get请求的结果:直接在浏览器中请求。NOTE:响应的数据都是Hello World!

-----------------------请求头中的值-------------------------
GET /devil/index.html HTTP/1.1
Host: 127.0.0.1:8181
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10.12; rv:51.0) Gecko/20100101 Firefox/51.0
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: zh-CN,zh;q=0.8,en-US;q=0.5,en;q=0.3
Accept-Encoding: gzip, deflate
Connection: keep-alive
Upgrade-Insecure-Requests: 1
{Upgrade-Insecure-Requests= 1, Accept-Language= zh-CN,zh;q=0.8,en-US;q=0.5,en;q=0.3, Accept-Encoding= gzip, deflate, Connection= keep-alive, Accept= text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8}
-----------------------BODY 中的值-------------------------
-----------------------READ END-------------------------
-----------------------PRINT END-------------------------

post请求的结果:用火狐浏览器的编辑重发,在body中输入的值是index=12345。NOTE:响应的数据都是Hello World!

-----------------------请求头中的值-------------------------
post /devil/index.html HTTP/1.1
Host: 127.0.0.1:8181
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10.12; rv:51.0) Gecko/20100101 Firefox/51.0
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8
Accept-Language: zh-CN,zh;q=0.8,en-US;q=0.5,en;q=0.3
Accept-Encoding: gzip, deflate
Connection: keep-alive
Upgrade-Insecure-Requests: 1
Content-Length: 11
Pragma: no-cache
Cache-Control: no-cache
{Upgrade-Insecure-Requests= 1, Accept-Language= zh-CN,zh;q=0.8,en-US;q=0.5,en;q=0.3, Content-Length= 11, Accept-Encoding= gzip, deflate, Connection= keep-alive, Accept= text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8, Cache-Control= no-cache, Pragma= no-cache}
-----------------------BODY 中的值-------------------------
11
index=12345
-----------------------READ END-------------------------
-----------------------PRINT END-------------------------
那么我们现在就可以理解这个http协议,其是它就是人们定义一个通信格式和标准。就像语言一样,通过同一个语言来描述一件事物,那么对方就可以知道你想表达的意思,并用相同的语言和你交流(响应)。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值