DNS预解析和优化

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: DNS 的作用是将域名解析为 IP 地址,解析的过程是耗时的,转化后会做本地缓存,我们的优化的目标主要是针对用户第一次访问站点的时候陷入长时间白屏的问题。

简单来说,DNS 的作用是将域名解析为 IP 地址,解析的过程是耗时的,转化后会做本地缓存,我们的优化的目标主要是针对用户第一次访问站点的时候陷入长时间白屏的问题。

DNS 解析可以分为两类,第一类是页面 DNS 解析,当用户输入 url 地址后(比如是第一次访问站点),便开始页面 DNS 解析,这个过程是省不了的,因为你无法在用户访问站点之前就让他提前把 DNS 解析好;

第二类是其他资源的 DNS 解析,在浏览器解析 html 的时候,会遇到一些 script 元素、link 元素,此时会暂停 html 的解析,转而加载 JS,里面就包含了 DNS 解析,这个过程是耗时的,会阻塞浏览器渲染主线程,所以该如何进行优化呢?答案是采用 DNS 预解析!

一、DNS预解析是什么

DNS预解析dns-prefetch )是前端网络性能优化的一种措施,它根据浏览器定义的规则,提前解析之后可能会用到的域名,使解析结果缓存到系统缓存中,缩短DNS解析时间,进而提高网站的访问速度。

DNS预解析能够让浏览器在用户访问链接之前解析域名,其范围包括文档的所有链接,包括图片、CSS、JS;域名解析后,如果用户确实访问该域名,那么DNS解析时间将不会有延迟。因为预读取会在后台执行,所以DNS很可能在链接对应的东西出现之前就已经解析完毕,这能够减少用户点击链接时的延迟。

二、DNS预解析的原理

当浏览器访问一个域名的时候,需要解析一次 DNS,获得对应域名的 ip 地址;在解析过程中,按照如下的顺序逐步读取缓存,直到拿到IP地址:

  • 浏览器缓存
  • 系统缓存
  • 路由器缓存
  • ISP(运营商)DNS缓存
  • 根域名服务器
  • 顶级域名服务器
  • 主域名服务器

dns-prefetch 就是在将解析后的IP缓存在系统中;这样就有效地缩短了 DNS 解析时间。因为在本地操作系统做了 DNS 缓存,使得 DNS 在解析的过程中,提前在系统缓存中找到了对应 IP;这样一来,后续的解析步骤就不用执行了,进而也就缩短了 DNS 解析时间。

假如浏览器首次将一个域名解析为 IP 地址,并缓存至操作系统,那么下一次 DNS 解析时间可以低至 0-1ms;倘若结果不缓存在系统,那么就需要读取路由器的缓存,进而后续的解析时间最小也要约 15ms

如果路由器缓存也不存在,则需要读取 ISP(运营商)DNS缓存,一般像 taobao.combaidu.com 这些常见的域名,读取ISP(运营商)DNS 缓存需要的时间在 80-120ms,如果是不常见的域名,平均需要 200-300ms

那也就是说,dns-prefetch 可以给 DNS 解析过程带来 15-300ms 的提升,尤其是一些大量引用很多其他域名资源的网站,提升效果就更加明显了。

三、如何开启DNS预解析

在 HTML 的 head 部分添加以下代码来启用 DNS 预解析,href 属性指定了需要预解析的主机名:

js

复制代码

<link rel="dns-prefetch" href="//baidu.com">

需要注意的是,dns-prefetch 仅对跨域域上的 DNS 查找有效,因此请避免使用它来指向相同域

HTTP 页面下所有的 a 标签的 href 都会自动去启用 DNS Prefetch,也就是说,你网页的 a 标签 href 带的域名,是不需要在 head 里面加上 link 手动设置的。

四、开启DNS预解析的前后对比

准备了一个例子,使用了掘金 logo 的地址,预解析该图片所在域名 dns,然后在点击切换按钮的时候把当前图片替换成掘金 logo

点击切换按钮,查看 network 该图片请求的 Timing,发现少了一个 DNS 解析阶段:

再来看看不开启预解析 dns 的情况:

结果如下,多了 dns 解析这一阶段:

继续点击切换,请求如下,dns 解析、建立连接时间(Socket) + SSL认证时间都没有:

其实就等于 preconnect(dns 解析、建立连接时间(Socket) + SSL认证时间都提前):

js

复制代码

  <link rel="preconnect" href="https://lf-cdn-tos.bytescm.com/">

五、项目中使用DNS预解析

在项目中我们可能会遇到一个问题,就是很多地方使用到了第三方的外链,比如图片、CSS、JS,由于项目是团队开发,有时候还不知道项目哪些地方引用了第三方的外链,所以我们不可能通过 link 标签的形式将这些第三方外链一个个引入并开启 DNS 预解析。

这个时候需要写一个插件去帮我们查找项目中所有的引入的第三方外链,在网上搜了下找不到满足需求的插件,只能自己写了,由于 Vite 项目使用 rollup 打包的,而 Vue-cli 项目用的是 webpack,不同的工具打包机制是不一样的,所以不采用插件的形式,在运行打包命令的时候运行一段代码,去修改打包的结果,这里以 Vite 工程为例:

js

复制代码

// package.json"scripts": {
    "build": "vite bulid && node ./scripts/dns-prefetch.js"}

具体的代码如下,简单来说就是,遍历打包后的 dist 目录中的所有 HTML、JS、CSS 文件,将所有外链的域名存起来,然后在 dist 目录下 index.html 文件的 head 标签中依次插入 link 标签,同时开启 DNS 预解析:

js

复制代码

// dns-prefetch.jsconst fs = require('fs')
const path = require('path')
const { parse } = require('node-html-parser')
const { glob } = require('glob')
const urlRegex = require('url-regex')

// 获取外部链接的正则表达式const urlPattern = /(https?:\/\/[^/]*)/iconst urls = newSet()

// 遍历dist目录中的所有HTML、JS、CSS文件asyncfunctionsearchDomin() {
    const files = awaitglob('dist/**/*.{html,css,js}')    for (const file of files) {        const source = fs.readFileSync(file, 'utf-8')        const matches = source.match(urlRegex({ strict: true }))        if (matches) {            matches.forEach((url) => {                const match = url.match(urlPattern)                if (match && match[1]) {                    urls.add(match[1])                }            })        }    }}// 在index.html文件<head>标签中插入link标签asyncfunctioninsertLinks() {
    const files = awaitglob('dist/**/*.html')    const links = [...urls].map((url) =>`<link rel="dns-prefetch" href="${url}" />`).join('\n')
    
    for (const file of files) {        const html = fs.readFileSync(file, 'utf-8')        const root = parse(html)        const head = root.querySelector('head')        head.insertAdjacentHTML('afterbegin', links)        fs.writeFileSync(file, root.toString())    }}asyncfunctionmain() {
    awaitsearchDomin()    awaitinsertLinks()}main()


作者:前端掘金者H

链接:https://juejin.cn/post/7285915718667124771


目录
相关文章
|
13天前
|
自然语言处理 数据可视化 API
淘宝商品评论 API 接口:深度解析用户评论,优化产品与服务
淘宝是领先的中国电商平台,其API为开发者提供商品信息、交易记录及用户评价等数据访问服务。对于获授权的开发者和商家,可通过申请API权限、获取并解析评论数据来进行情感分析和统计,进而优化产品设计、提升服务质量、增强用户互动及调整营销策略。未授权用户可能受限于数据访问。
|
18天前
|
网络协议 Linux Docker
在Linux中,如何指定dns服务器,来解析某个域名?
在Linux中,如何指定dns服务器,来解析某个域名?
|
7天前
|
网络协议
DNS正向解析实现
文章介绍了DNS正向解析的实现,包括资源记录的定义、配置区域解析记录的步骤,并通过实际操作展示了如何为"yinzhengjie.com"域名配置DNS解析记录。
18 2
DNS正向解析实现
|
5天前
|
XML 存储 数据格式
RAG效果优化:高质量文档解析详解
本文关于如何将非结构化数据(如PDF和Word文档)转换为结构化数据,以便于RAG(Retrieval-Augmented Generation)系统使用。
|
7天前
|
机器学习/深度学习 数据采集 存储
一文读懂蒙特卡洛算法:从概率模拟到机器学习模型优化的全方位解析
蒙特卡洛方法起源于1945年科学家斯坦尼斯劳·乌拉姆对纸牌游戏中概率问题的思考,与约翰·冯·诺依曼共同奠定了该方法的理论基础。该方法通过模拟大量随机场景来近似复杂问题的解,因命名灵感源自蒙特卡洛赌场。如今,蒙特卡洛方法广泛应用于机器学习领域,尤其在超参数调优、贝叶斯滤波等方面表现出色。通过随机采样超参数空间,蒙特卡洛方法能够高效地找到优质组合,适用于处理高维度、非线性问题。本文通过实例展示了蒙特卡洛方法在估算圆周率π和优化机器学习模型中的应用,并对比了其与网格搜索方法的性能。
74 1
|
18天前
|
域名解析 存储 缓存
在Linux中,DNS进行域名解析的过程是什么?
在Linux中,DNS进行域名解析的过程是什么?
|
10天前
|
开发者 图形学 iOS开发
掌握Unity的跨平台部署与发布秘籍,让你的游戏作品在多个平台上大放异彩——从基础设置到高级优化,深入解析一站式游戏开发解决方案的每一个细节,带你领略高效发布流程的魅力所在
【8月更文挑战第31天】跨平台游戏开发是当今游戏产业的热点,尤其在移动设备普及的背景下更为重要。作为领先的游戏开发引擎,Unity以其卓越的跨平台支持能力脱颖而出,能够将游戏轻松部署至iOS、Android、PC、Mac、Web及游戏主机等多个平台。本文通过杂文形式探讨Unity在各平台的部署与发布策略,并提供具体实例,涵盖项目设置、性能优化、打包流程及发布前准备等关键环节,助力开发者充分利用Unity的强大功能,实现多平台游戏开发。
25 0
|
10天前
|
开发者 图形学 UED
深度解析Unity游戏开发中的性能瓶颈与优化方案:从资源管理到代码执行,全方位提升你的游戏流畅度,让玩家体验飞跃性的顺滑——不止是技巧,更是艺术的追求
【8月更文挑战第31天】《Unity性能优化实战:让你的游戏流畅如飞》详细介绍了Unity游戏性能优化的关键技巧,涵盖资源管理、代码优化、场景管理和内存管理等方面。通过具体示例,如纹理打包、异步加载、协程使用及LOD技术,帮助开发者打造高效流畅的游戏体验。文中提供了实用代码片段,助力减少内存消耗、提升渲染效率,确保游戏运行丝滑顺畅。性能优化是一个持续过程,需不断测试调整以达最佳效果。
23 0
|
10天前
|
Java 数据库 API
JSF与JPA的史诗级联盟:如何编织数据持久化的华丽织锦,重塑Web应用的荣耀
【8月更文挑战第31天】JavaServer Faces (JSF) 和 Java Persistence API (JPA) 分别是构建Java Web应用的用户界面组件框架和持久化标准。结合使用JSF与JPA,能够打造强大的数据驱动Web应用。首先,通过定义实体类(如`User`)和配置`persistence.xml`来设置JPA环境。然后,在JSF中利用Managed Bean(如`UserBean`)管理业务逻辑,通过`EntityManager`执行数据持久化操作。
17 0
|
10天前
|
开发者 Java
Play Framework深度解析:依赖注入的神秘力量,如何助力Web应用架构优化?答案即将揭晓!
【8月更文挑战第31天】依赖注入(DI)是现代软件开发的关键技术,用于分离对象创建与依赖关系,提升代码的可维护性和可测试性。Play Framework是一款高性能Java Web框架,内置了基于Google Guice的DI支持。本文探讨Play Framework中DI的最佳实践,包括定义组件、构造函数注入、字段注入以及作用域控制和自定义绑定等高级特性,帮助开发者轻松构建结构清晰、可维护性高的Web应用。
21 0

相关产品

  • 云解析DNS
  • 推荐镜像

    更多
    下一篇
    DDNS