1.背景介绍
当你在浏览器地址栏输入一个URL后回车,将会发生的事情?这是一道经典的面试题,同时也是一道复杂的题目,涉及到很多东西,不同的软件开发者对于此道问题有不同的答案,对于其中的某一点也能无限深究,今天我们就来侧重于web前端来看一下究竟发生了什么。
2.知识剖析
基本流程
查询ip地址
建立tcp连接,接入服务器
浏览器发起http请求
服务器后台操作并做出响应
浏览器渲染页面
3.常见问题
1)URL是什么
统一资源定位符是对可以从互联网上得到的资源的位置和访问方法的一种简洁的表示,是互联网上标准资源的地址。互联网上的每个文件都有一个唯一的URL,它包含的信息指出文件的位置以及浏览器应该怎么处理它。
4.解决方案
1)查询IP地址
浏览器解析出url中的域名
查询浏览器的DNS缓存
浏览器中没有DNS缓存,则查找本地客户端hosts文件有无对应的ip地址
hosts中无,则查找本地DNS服务器(运营商提供的DNS服务器)有无对应的DNS缓存
若本地DNS没有DNS缓存,则向根服务器查询,进行递归查找
递归查找从顶级域名开始(如.com),一步步缩小范围,最终客户端取得ip地址
2)TCP连接与HTTP连接
http协议建立在tcp协议之上,http请求前,需先进行tcp连接,形成客户端到服务器的稳定的通道。俗称TCP的三次握手
tcp连接完成后,http请求开始,请求有多种方式,常见的有get,post等
http请求包含请求头,也可能包含请求体两部分,请求头中包含我们希望对请求文件的操作的信息,请求体中包含传递给后台的参数。
服务器收到http请求后,后台开始工作,如负载平衡,跨域等,这里就是后端的工作了。
文件处理完毕,生成响应数据包,响应也包含两部分,响应头和相应体,响应体就是我们所请求的文件
经过网络传输,文件被下载到本地客户端,客户端开始加载
3)HTML渲染
客户端浏览器加载了html文件后,由上到下解析html为DOM树(DOM Tree)
遇到css文件,css中的url发起http请求。
这是第二次http请求,由于http1.1协议增加了Connection: keep-alive声明,故tcp连接不会关闭,可以复用。
http连接是无状态连接,客户端与服务器端需要重新发起请求--响应
在请求css的过程中,解析器继续解析html,然后到了script标签。
由于script可能会改变DOM结构,故解析器停止生成DOM树,解析器被js阻塞,等待js文件发起http请求,然后加载。这是第三次http请求。js执行完成后解析器继续解析。
由于css文件可能会影响js文件的执行结果,因此需等css文件加载完成后再执行。
浏览器收到css文件后,开始解析css文件为CSSOM树(CSS Rule Tree)。
CSSOM树生成后,DOM Tree与CSS Rule Tree结合生成渲染树(Render Tree),
Render Tree会被css文件阻塞,渲染树生成后,先布局,绘制渲染树中节点的属性(位置,宽度,大小等),然后渲染,页面就会呈现信息。
继续边解析边渲染,遇到了另一个js文件,js文件执行后改变了DOM树,渲染树从被改变的dom开始再次渲染
继续向下渲染,碰到一个img标签,浏览器发起http请求,不会等待img加载完成,继续向下渲染,之后再重新渲染此部分。
DOM树遇到html结束标签,停止解析,进而渲染结束
5.编码实战
6.扩展思考
1)有那些网站优化的方法?
减少DNS查询:将服务器域名的ip信息加入本地host文件
减少http请求数量,对于图片使用雪碧图,对于html文件和css文件,js文件分别进行合并操作。
减少下载时间:压缩图片,使用压缩应用压缩文档中的空格,删除文件多余的语句和注释,创造自己的js精简库和精简框架,使用本地浏览器缓存。
提前渲染开始时间:将css链接放在html头部
减轻解析器的阻塞:将js链接放在body尾部
使用打包工具gulp webpack开启Gzip压缩文件
提问
1.TCP/IP协议 的3次握手
所谓三次握手(Three-Way Handshake)即建立TCP连接,就是指建立一个TCP连接时,需要客户端和服务端总共发送3个包以确认连接的建立。在socket编程中,这一过程由客户端执行connect来触发,整个流程如下图所示:
(1)第一次握手:Client将标志位SYN置为1,随机产生一个值seq=J,并将该数据包发送给Server,Client进入SYN_SENT状态,等待Server确认。
(2)第二次握手:Server收到数据包后由标志位SYN=1知道Client请求建立连接,Server将标志位SYN和ACK都置为1,ack=J+1,随机产生一个值seq=K,并将该数据包发送给Client以确认连接请求,Server进入SYN_RCVD状态。
(3)第三次握手:Client收到确认后,检查ack是否为J+1,ACK是否为1,如果正确则将标志位ACK置为1,ack=K+1,并将该数据包发送给Server,Server检查ack是否为K+1,ACK是否为1,如果正确则连接建立成功,Client和Server进入ESTABLISHED状态,完成三次握手,随后Client与Server之间可以开始传输数据了。
7.参考文献
参考一:当我们输入URI,按下回车发生了什么?
参考二:当你在浏览器地址栏输入一个URL后回车,将会发生的事情?