手写一个简单的Http服务器

Posted 既来之,则安之!

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了手写一个简单的Http服务器相关的知识,希望对你有一定的参考价值。

 

 

什么是Http协议?

Http协议: 对浏览器客户端 和 服务器端 之间数据传输的格式规范

Http是一个属于应用层的面向对象的协议,由于其简捷、快速的方式,适用于分布式超媒体信息系统。它于1990年提出,经过几年的使用与发展,得到不断地完善和扩展。目前在WWW中使用的是HTTP/1.0的第六版,HTTP/1.1的规范化工作正在进行之中,而且HTTP-NG(Next Generation of HTTP)的建议已经提出。
HTTP协议的主要特点可概括如下:
1.支持客户/服务器模式。
2.简单快速:客户向服务器请求服务时,只需传送请求方法和路径。请求方法常用的有GET、HEAD、POST。每种方法规定了客户与服务器联系的类型不同。由于HTTP协议简单,使得HTTP服务器的程序规模小,因而通信速度很快。
3.灵活:HTTP允许传输任意类型的数据对象。正在传输的类型由Content-Type加以标记。
4.无连接:无连接的含义是限制每次连接只处理一个请求。服务器处理完客户的请求,并收到客户的应答后,即断开连接。采用这种方式可以节省传输时间。
5.无状态:HTTP协议是无状态协议。无状态是指协议对于事务处理没有记忆能力。缺少状态意味着如果后续处理需要前面的信息,则它必须重传,这样可能导致每次连接传送的数据量增大。另一方面,在服务器不需要先前信息时它的应答就较快。

Http是使用Socket中的TCP协议实现,基于请求(客户端)、响应(服务器端)

实例:http://127.0.0.1:8080/项目路径

Http协议分为请求报文和响应报文

请求模式 Get、Post、Delete

默认Http请求参数(请求报文)

GET / HTTP/1.1

Host: 127.0.0.1:8080

Connection: keep-alive

Cache-Control: max-age=0

Upgrade-Insecure-Requests: 1

User-Agent: Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (Khtml, like Gecko) Chrome/53.0.2783.4 Safari/537.36

Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8

Accept-Encoding: gzip, deflate, sdch

Accept-Language: zh-CN,zh;q=0.8

Cookie: Hm_lvt_b43c1c82b09cda6b125e6981fbde442c=1487946887,1487950668,1488114252; CNZZDATA1259976290=1895288300-1478441025-%7C1488116464

 

 

Http响应报文

HTTP响应报文与HTTP请求报文是对应的,也是分为三个部分。

  1、响应行

  2、响应头

  3、响应体

 

项目结构图:

上代码

封装响应的数据类
package cn.liuzhiw.server;

import java.io.BufferedWriter;
import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.util.Date;

/**
 * 封装响应的数据类
 * Created by 刘志威 on 2018/4/24.
 */
public class Response {
    //空格
    private final String BLANK="  ";
    //换行
    private final String BR="\\r\\n";
    //显示Http响应头
    private StringBuffer responseData;
    //封装网页信息
    private StringBuffer responseInfo;
    //字符缓冲输出流
    private BufferedWriter bufferedWriter;
    //响应的长度
    private long len;
    //初始化配置
    public Response(OutputStream outputStream){
        responseData=new StringBuffer();
        responseInfo=new StringBuffer();
        bufferedWriter=new BufferedWriter(new OutputStreamWriter(outputStream));
    }
    //创建头部信息
    private void createHeadInfo(int code){
        responseData.append("HTTP/1.1");
        responseData.append(BLANK);
        responseData.append(code + "").append(BLANK);
        switch (code){
            case 200:
                responseData.append("ok");
                break;
            case 404:
                responseData.append("not found");
                break;
            case 500:
                responseData.append("server error");
                break;
            default:
                break;
        }
        responseData.append("Date:").append(new Date()).append(BR);
        responseData.append("Content-Type: text/html;charset=UTF-8").append(BR);
        responseData.append("Content-Length:").append(len).append(BR);
        responseData.append(BR);
    }

    /**
     * 填充页面类容
     * @param str
     * @return
     */
    public Response print(String str) {
        responseInfo.append(str);
        len += str.getBytes().length;
        return this;
    }
    /**
     * 发送数据
     * @throws IOException
     */
    public void connect(int code) throws IOException {
        createHeadInfo(code);
        bufferedWriter.append(responseData.toString());
        bufferedWriter.append(responseInfo.toString());
        //刷新
        bufferedWriter.flush();
        //关闭
        bufferedWriter.close();
    }
}

  

服务器核心配置类
package cn.liuzhiw.server;

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.Date;

/**
 * 服务器核心配置类
 * Created by 刘志威 on 2018/4/24.
 */
public class Server4 {
    //socket服务端
    private ServerSocket serverSocket;
    //端口号
    private final int PORT=8080;
    /**
     * 启动服务器
     */
    public void start() throws IOException {
        System.out.println("服务器启动成功.........");
        //监听服务器
        serverSocket=new ServerSocket(PORT);
        //接受客户端数据
        acceptClient();
        //关闭链接
        closeServer();

    }

    /**
     * 接受客户端数据
     * @throws IOException
     */
    public void acceptClient() throws IOException {
        //拿到socket客户端对象
        Socket accept = serverSocket.accept();
        //拿到输入流
        InputStream inputStream = accept.getInputStream();
        byte []data=new byte[20480];
        int read = inputStream.read(data);
        String message = new String(data, 0, read);
        System.out.println("客户端发送来的数据:"+message);
        //响应回去数据
        Response response=new Response(accept.getOutputStream());
        response.print("<html><head><title>响应页面</title></head><body>响应页面成功</body></html>");
        response.connect(200);
    }
    /**
     * 关闭链接
     * @throws IOException
     */
    public void closeServer() throws IOException {
        if (serverSocket!=null){
            serverSocket.close();
        }
    }
}

  

服务器启动入口
package cn.liuzhiw.app;

import cn.liuzhiw.server.Server4;

import java.io.IOException;

/**
 * 服务器启动入口
 * Created by 刘志威 on 2018/4/24.
 */
public class App {
    public static void main(String[] args) throws IOException {
        Server4 server=new Server4();
        server.start();
    }
}

  html请求的页面

<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Http服务器post方式提交参数</title>
</head>
<body>
    <pre>
     method:请求方式 get/post
     get:数据量小,安全性不高  默认方式
     post:量大,安全性相对高
     action:请求的服务器路径
     id:编号:前端区分唯一,js使用
     name:名称后端(服务器)区分唯一性,获取值
           只要提交数据给后台,必须存在name;
    
    </pre>
   <form action="http://127.0.0.1:8080/index" method="post">
    用户名称:<input type="text" name="uname" id="uname">
    用户面:<input type="password" name="pwd" id="pwd">
   <input type="submit">
   </form>
</body>
</html>

页面效果和后台效果

 

以上是关于手写一个简单的Http服务器的主要内容,如果未能解决你的问题,请参考以下文章

代码片段 - Golang 实现简单的 Web 服务器

前端面试题之手写promise

利用Java手写简单的httpserver

资深架构师带你通过手写代码实现服务的注册与发现~ 附代码示例链接!

ajax简单手写了一个猜拳游戏

简单的方法来分享/讨论/协作的代码片段?