浅谈Socket 实现Http服务器

目标:更好的理解 Apache、Tomcat 等软件的工作模式

之前做web应用一直是在本地装个Apache、Tomcat之类的软件,然后把做好的网页文件放在他们的工作目录下(如Apache的htdocs),然后打开浏览器输入127.0.0.1或localhost就可以直接访问了,好神奇,可是为什么,怎么实现的呢,早就知道有Socket(套接字)这个东西,可之前就是没有把这两方面结合起来,今天我们就一起来看一看这究竟是为什么。

先看下Http请求命令长啥样:

  1. 服务器代码
public static void main(String[] args) {
    try {
            ServerSocket serverSocket = new ServerSocket(80);
            System.out.println("正在等待情书中...");
            Socket socket = serverSocket.accept();
            System.out.println("收到情书,我要开始解析!");
                
            String line3 = "";   
            InputStream inputStream3 = socket.getInputStream();   
                BufferedReader reader3 = new BufferedReader(new InputStreamReader(inputStream3));
                while ( (line3 = reader3.readLine()) != null ){
                    System.out.println(line3);   
                }  
          } catch (Exception e) {
             e.printStackTrace();
          }
}
  1. 客户端访问
    浏览器输入:http://127.0.0.1/html/index3.html
    服务端的打印结果如下图:

    注意看第一行:
    GET /html/index3.html HTTP/1.1
    • 方法为GET
    • 请求的文件为/html/index3.html
    • Http 版本为1.1

服务器最终代码

public static void main(String[] args) {
        // TODO Auto-generated method stub
        String base_url = "C:/Users/Administrator/Downloads/";
        while(true){
            try {
                ServerSocket serverSocket = new ServerSocket(80);
                System.out.println("正在等待情书中...");
                Socket socket = serverSocket.accept();
                System.out.println("收到情书,我要开始解析!");
                
                //得到想访问的文件名
                InputStream inputStream = socket.getInputStream();
                BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
                String line = reader.readLine();
                String url = line.substring(5, line.indexOf("HTTP") - 1);
                
                //生成输出流
                FileInputStream fileInputStream = new FileInputStream(base_url + url);
                OutputStream outputStream = socket.getOutputStream();
                byte[] buffer = new byte[4 * 1024];
                int len = 0;
                while ((len = fileInputStream.read(buffer)) >0) {
                    outputStream.write(buffer, 0, len); 
                }
                outputStream.flush();
                
                serverSocket.close();  
                fileInputStream.close();
                inputStream.close();
                outputStream.close();
                
                
                
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        

    }

注意:socket 要正常关闭,不然会出现异常。
服务器的目录如下:
根目录


html目录

多媒体目录

这样便可进行Http的访问了,

  1. 访问Html 文件
    浏览器输入 http://127.0.0.1/html/index3.html
  2. 访问视频文件
    浏览器输入http://127.0.0.1/media/Shakira-LaLaLa.mp4
  3. 访问图片文件
    浏览器输入 http://127.0.0.1/media/gif7.gif
  4. 访问音频文件
    浏览器输入 http://127.0.0.1/media/Button47.wav

边角料( 转 )

  • Q:什么是端口?
    A:这是一个比较抽象的概念,是为了进程间通信,每一个进程只能占用一个端口,也就是说多个进程绝不能同时占用一个端口

  • Q:既然多个进程不能同时占用一个端口,那么咱们常说的web服务默认使用的是80端口,我电脑有三个浏览器,谷歌,360,IE他们却可以同时上网,这不是端口冲突了吗?
    A:常说的web服务使用80端口指的是服务器监听web请求的端口,是服务器,不是你自己的客户机。一般来说,一个应用程序打开后访问网络本地操作系统为其分配的端口号是随机的,所以三个浏览器虽然同时接收web服务器的回复报文,由于他们三个各自占用的端口不一样,所以不会产生冲突。

  • Q:非要使用80端口吗?
    A:不一定。我们刚才在编程的时候确实使用的是80端口,所以我们在浏览器中输入127.0.0.1/index.html,浏览器会默认认为我们会向127.0.0.1主机的80号端口发送请求。但是,这个80端口号只是默认的而已,我们完全可以自己改掉,比如在java代码里把服务器端的ServerSocket改成 ServerSocket serverSocket = new ServerSocket(3456); 这时候我们在浏览器中就要输入 127.0.0.1:3456/index.html 了,效果是一样的,可以浅尝辄止一下。

  • Q:谁是客户端,谁是服务器?
    A:咱们只有一台电脑,这台电脑既充当着客户端的角色,又充当着服务器的角色。当浏览器请求网页时,它是客户端;当80端口收到请求报文并应答时,它就是服务器。实在不理解,就想想什么是自恋吧

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容