Java网络编程 - 浅析web服务器与浏览器的实现原理_java中浏览器和服务器的关系

独家面经总结,超级精彩

本人面试腾讯,阿里,百度等企业总结下来的面试经历,都是真实的,分享给大家!

image

image

image

image

Java面试准备

准确的说这里又分为两部分:

  1. Java刷题
  2. 算法刷题

Java刷题:此份文档详细记录了千道面试题与详解;

image

image

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

一个HTTP协议的请求中,通常主要包含三个部分:

  • 方法/统一资源标示符(URI)/协议/版本
  • 请求标头
  • 实体主体

其中方法也就是所谓的get/post之类的请求方法,统一资源标示符也就是要访问的目标资源的路径,包括协议及协议版本,这些信息被放在请求的第一行。

随后,紧接着的便是请求标头;请求标头通常包含了与客户端环境及请求实体主体相关的有用信息。

最后,在标头与实体主体之间是一个空行。它对于HTTP请求格式是很重要的,空行告诉HTTP服务器,实体主体从这里开始。

前面已经说过了,我们这里想要研究的,是WEB服务器的基本实现原理。

那么我们自然想要自己来实现一下所谓的WEB服务器,我们已经知道了:

所谓的B/S结构,实际上就是客户端与服务器之间基于HTTP协议的网络通信。

那么,肯定是离不开socket与io的,所以我们可以简单的模拟一个最简易功能的山寨浏览器:

public class MyTomcat {
	public static void main(String[] args) {
		try {
			ServerSocket tomcat = new ServerSocket(9090);
			System.out.println("服务器启动");
			//
			Socket s = tomcat.accept();
			//
			byte[] buf = new byte[1024];
			InputStream in = s.getInputStream();
			//
			int length = in.read(buf);
			String request = new String(buf,0,length);
			//
			System.out.println(request);

		} catch (IOException e) {
			// TODO Auto-generated catch block
			e.printStackTrace();
		}
	}
}

这次我们在通过对应的URL在浏览器中对我们的山寨服务器进行访问,得到的输出结果是:

通过成果我们看到,我们已经成功的简单山寨了一下tomcat。

不过这里需要注意的是,我们自己山寨的tomcat服务器当中,之所以也成功的输出了Http协议的请求体,是因为:

我们是通过web浏览器进行访问的,如果通过普通的socket进行对serversocket的连接访问,是没有这些请求信息的。

因为我们前面已经说过了,web浏览器与服务器之间的通信必须遵循Http协议。

所以,我们日常生活中使用的web浏览器,会自动的为我们的请求进行基于http协议的包装。

但是,因为我们已经了解了原理,所以我们也可以自己模拟一下浏览器过过瘾:

//山寨浏览器
public class MyBrowser {

    public static void main(String[] args) {
        try {
            Socket browser = new Socket("192.168.1.102", 9090);
            PrintWriter pw = new PrintWriter(browser.getOutputStream(),true);
            // 封装请求第一行
            pw.println("GET/ HTTP/1.1");
            // 封装请求头
            pw.println("User-Agent: Java/1.6.0_13");
            pw.println("Host: 192.168.1.102:9090");
            pw
                    .println("Accept: text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2");
            pw.println("Connection: keep-alive");
            // 空行
            pw.println();
            // 封装实体主体
            pw.println("UserName=zhangsan&Age=17");
            // 写入完毕
            browser.shutdownOutput();
            
            
            // 接受服务器返回信息,
            InputStream in = browser.getInputStream();
            //
            int length = 0;
            StringBuffer request = new StringBuffer();
            byte[] buf = new byte[1024];
            //
            while ((length = in.read(buf)) != -1) {
                String line = new String(buf, 0, length);
                request.append(line);
            }
            System.out.println(request);
            //browser.close();
        } catch (IOException e) {
            System.out.println("异常了,操!");
        }finally{
            
        }
    }
}

//修改后的山寨tomcat服务器
public class MyTomcat {
    public static void main(String[] args) {
        try {
            ServerSocket tomcat = new ServerSocket(9090);
            System.out.println("服务器启动");
            //
            Socket s = tomcat.accept();
            //
            byte[] buf = new byte[1024];
            InputStream in = s.getInputStream();
            //

            int length = 0;
            StringBuffer request = new StringBuffer();
            while ((length = in.read(buf)) != -1) {
                String line = new String(buf, 0, length);
                request.append(line);
            }
            //
            System.out.println("request:"+request);

            PrintWriter pw = new PrintWriter(s.getOutputStream(),true);
            pw.println("<html>");
            pw.println("<head>");
            pw.println("<title>LiveSession List</title>");
            pw.println("</head>");
            pw.println("<body>");
            pw.println("<p style=\"font-weight: bold;color: red;\">welcome to MyTomcat</p>");
            pw.println("</body>");
            s.close();
            tomcat.close();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

我们先启动服务器,然后运行浏览器模拟网页浏览的过程,首先看到服务器端收到的请求信息:

紧接着,服务器收到请求进行处理后,返回资源给浏览器,于是得到输出信息:

可以看到,我们在山寨浏览器当中得到的返回信息,实际上就是一个HTML文件的源码,

之所以我们的山寨浏览器中,这些信息仅仅是以纯文本形式显示,是因为我们的山寨浏览器不具备解析HTML语言的能力。

所以说,浏览器另外一个重要的功能其实就是:可以对超文本标记语言进行解析。而实际上,这也是浏览器开发的难点和重点。

上面这样的输出结果看上去显然不爽,所以说山寨货毕竟还是坑爹!

我们还是通过正规的WEB浏览器,来试着访问一下我们的山寨服务器,结果发现,效果帅多了:

而顺带一提的是,既然当浏览器向WEB服务器发起访问请求时,会封装有对应的HTTP请求体。

那么,对应的,当WEB服务器处理完浏览器请求,返回数据时,也会有对应的封装,就是所谓的HTTP响应体。

举例来说,假如我们将我们的山寨浏览器的代码进行修改,去连接真正的tomcat服务器:

public class MyBrowser {

	public static void main(String[] args) {
		try {
			Socket browser = new Socket("192.168.1.102", 8080);
			PrintWriter pw = new PrintWriter(browser.getOutputStream(),true);
			// 封装请求第一行
			pw.println("GET / HTTP/1.1");
			// 封装请求头
			pw.println("User-Agent: Java/1.6.0_13");
			pw.println("Host: 192.168.1.102:8080");
			pw
					.println("Accept: text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2");
			pw.println("Connection: keep-alive");
			// 空行
			pw.println();
			// 封装实体主体
			//pw.println("UserName=zhangsan&Age=17");
			// 写入完毕
			browser.shutdownOutput();
			
			
			// 接受服务器返回信息,
			InputStream in = browser.getInputStream();
			//
			int length = 0;
			StringBuffer request = new StringBuffer();
			byte[] buf = new byte[1024];
			//
			while ((length = in.read(buf)) != -1) {
				String line = new String(buf, 0, length);
				request.append(line);
			}
			System.out.println(request);
			//browser.close();
		} catch (IOException e) {
			System.out.println("异常了,操!");
		}finally{
			
		}
	}
}

运行程序,你将会发下如下的输出信息:

与HTTP请求类似,通常一个HTTP响应也包含三个部分:

  • 协议/响应码/状态描述:协议也就是指HTTP协议的信息,响应码是指代表该次请求的处理结果的码(例如常见的200、404、500),其实就是该次请求处理的响应描述
  • 响应标头:响应标头也包含与HTTP请求中的标头类似的有用信息。
  • 响应实体:通常也就是指响应本身的HTML内容。

与HTTP请求一样,响应表头与响应实体之间,也会使用一个空行进行分割,方便解读。

同时我们也可以发现,其实真正被解析显示在浏览器网页上的内容,其实只是响应实体的部分。

总结:绘上一张Kakfa架构思维大纲脑图(xmind)

image

其实关于Kafka,能问的问题实在是太多了,扒了几天,最终筛选出44问:基础篇17问、进阶篇15问、高级篇12问,个个直戳痛点,不知道如果你不着急看答案,又能答出几个呢?

若是对Kafka的知识还回忆不起来,不妨先看我手绘的知识总结脑图(xmind不能上传,文章里用的是图片版)进行整体架构的梳理

梳理了知识,刷完了面试,如若你还想进一步的深入学习解读kafka以及源码,那么接下来的这份《手写“kafka”》将会是个不错的选择。

  • Kafka入门

  • 为什么选择Kafka

  • Kafka的安装、管理和配置

  • Kafka的集群

  • 第一个Kafka程序

  • Kafka的生产者

  • Kafka的消费者

  • 深入理解Kafka

  • 可靠的数据传递

  • Spring和Kafka的整合

  • SpringBoot和Kafka的整合

  • Kafka实战之削峰填谷

  • 数据管道和流式处理(了解即可)

image

image

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

SpringBoot和Kafka的整合

  • Kafka实战之削峰填谷

  • 数据管道和流式处理(了解即可)

[外链图片转存中…(img-caEiQcCK-1715452090561)]

[外链图片转存中…(img-dgZQtX23-1715452090561)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

  • 18
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值