web前端面试高频考点——性能优化篇(手写防抖、手写节流、XXS攻击、XSRF攻击)

简介: web前端面试高频考点——性能优化篇(手写防抖、手写节流、XXS攻击、XSRF攻击)

运行环境

一、在哪里运行

  • 运行环境即浏览器(server 端有nodejs)

二、 网页是如何加载并渲染出来的

1、资源的形式

  • html 代码
  • 媒体文件,如图片、视频等
  • JavaScript CSS

2、加载过程

  • DNS 解析:域名 -> IP 地址
  • 浏览器根据 IP地址向服务端发起 http 请求
  • 服务器处理 http 请求,并返回给浏览器

3、渲染过程

根据 HTML 代码生成 DOM Tree(文档对象模型树)

根据 CSS 代码生成 CSSOM(CSS对象模型)

DOM 树和 CSSOM 可以并行构建

将 DOM Tree 和 CSSOM 整合形成 Render Tree(渲染树)

根据 Render Tree 渲染页面

遇到 <script> 则暂停渲染,优先加载并执行 JS 代码,完成再继续

直至把 Render Tree 渲染完成

4、window.onload 和 DOMContentLoaded 区别

  • window.onload:页面的全部资源加载完才会执行,包括图片、视频等
  • DOMContentLoaded:DOM 渲染完即可执行,此时图片、视频还可能没有加载完
    window.addEventListener('load', function() {
        // 页面的全部资源加载完才会执行,包括图片、视频等
    })
    document.addEventListener('DOMContentLoaded', function() {
        // DOM 渲染完即可执行,此时图片、视频还可能没有加载完
    })

二、性能优化

1、性能优化原则

  • 多使用内存、缓存或其他方法
  • 减少CPU 计算量,减少网络加载耗时
  • 用空间换时间

2、性能优化 — 让加载更快

减少资源体积:压缩代码

减少访问次数:合并代码,SSR 服务器端渲染,缓存

使用更快的网络:CDN(分布在不同区域的边缘节点服务器群组成的分布式网络)

3、性能优化 — 让渲染更快

CSS 放在 head,JS 放在 body 最下面

尽早开始执行 JS,用 DOMContentLoaded 触发

懒加载(图片懒加载,上滑加载更多)

对 DOM 查询进行缓存

频繁 DOM 操作,合并到一起插入 DOM 结构

节流 throttle 防抖 debounce

4、前端性能优化示例

资源合并:

    <script src="a.js"></script>
    <script src="b.js"></script>
    <script src="c.js"></script>
    // 合并为
    <script src="abc.js"></script>

缓存:

  • 静态资源加 hash 后缀,根据文件内容计算 hash(webpack中的配置)
  • 文件内容不变,则 hash 不变,则 url 不变
  • url 和文件不变,则会自动触发 http 缓存机制,返回 304
  • CDN:

SSR:

  • 服务器端渲染:将网页和数据一起加载,一起渲染
  • 非 SSR(前后端分离):先加载网页,再加载数据,再渲染数据
  • 早先的 JSP ASP PHP,现在的 vue react SSR

懒加载:

先给一个容易加载的图片,等加载完成后把真正的图片展现出来

    <img id="img1" src="preview.png" data-realsrc="abc.png">
  let img1 = document.getElementById('img1')
    img1.src = img1.getAttribute('data-realsrc')

缓存 DOM 查询:

5685cfcd9245456991d9bfcf681256f1.png

多个 DOM 操作一起插入到 DOM 结构:

f3c19cf6e99c47f780bbdf26ba9aa19f.png

尽早开始 JS 执行:

    window.addEventListener('load', function() {
        // 页面的全部资源加载完才会执行,包括图片、视频等
    })
    document.addEventListener('DOMContentLoaded', function() {
        // DOM 渲染完即可执行,此时图片、视频还可能没有加载完
    })

5、防抖 debounce

  • 监听一个输入框的,文字变化后触发 change 事件
  • 直接用 keyup 事件,则会频繁触发 change 事件
  • 防抖:用户输入结束或暂停时,才会触发 change 事件

示例:input 框的防抖

    <input type="text" id="input1">
    let input1 = document.getElementById('input1')
    let timer = null
    input1.addEventListener('keyup', function(value) {
        if(timer) {
            clearTimeout(timer)
        }
        timer = setTimeout(() => {
            // 模拟触发 change 事件
            console.log(input1.value) 
            // 清空定时器
            timer = null
        }, 500)
    })

7baf4d4b979e4a49905666dfae4c96d4.png

封装防抖 debounce 函数:

    <input type="text" id="input1">
    let input1 = document.getElementById('input1')
  // 防抖
    function debounce(fn, delay = 500) {
        // timer 是闭包中的
        let timer = null
        return function() {
            if(timer) {
                clearTimeout(timer)
            }
            timer = setTimeout(() => {
                fn.apply(this, arguments)
                timer = null
            }, delay)
        }
    }
    input1.addEventListener('keyup', debounce(() => {
        console.log(input1.value)
    }, 1000))

78391ca6b52f4a539e197faa56a2b906.png

6、节流 throttle

  • 拖拽一个元素时,要随时拿到该元素被拖拽的位置
  • 直接用 drag 事件,则会频繁触发,很容易导致卡顿
  • 节流:无论拖拽速度多快,都会每隔 xxxms 触发一次

示例:拖拽一个 div

    #div1 {
        border: 1px solid #ccc;
        width: 200px;
        height: 100px;
    }
  <div id="div1" draggable="true">可拖拽</div>
  const div1 = document.getElementById('div1')
    let timer = null
    div1.addEventListener('drag', function (event) {
        if(timer) {
            return 
        }
        timer = setTimeout(() => {
            console.log(event.offsetX, event.offsetY)
            timer = null
        }, 100)
    })

封装节流 throttle 函数:

    #div1 {
        border: 1px solid #ccc;
        width: 200px;
        height: 100px;
    }
    <div id="div1" draggable="true">可拖拽</div>
  const div1 = document.getElementById('div1')
    function throttle(fn, delay = 100) {
        let timer = null
        return function() {
            if(timer) {
                return
            }
            timer = setTimeout(() => {
                fn.apply(this, arguments)
                timer = null
            }, delay)
        }
    }
    div1.addEventListener('drag', throttle(function (event) {
        console.log(event.offsetX, event.offsetY)
    },))

三、常见的 web 前端攻击方式

  • XSS 跨站请求攻击
  • XSRF 跨站请求伪造

1、XXS 攻击

  • 一个博客网站,我发表一篇博客,其中嵌入 <script></script> 脚本
  • 脚本内容:获取 cookie,发送到我的服务器(服务器配合跨域)
  • 发表这篇博客,有人查看它,我轻松收割访问者的 cookie

2、XXS 预防

  • 替换特殊字符,如 < 变为 &lt; > 变为 &gt;
  • <script> 变为 &lt;script&gt; 直接显示,而不会作为脚本执行
  • 前端要替换,后端也要替换,都做更保险

示例:脚本不会执行,直接在页面显示

  &lt;script&gt;alert(document.cookie)&lt;script&gt;

2b3aeb91ef3c49c1b8945987b97f48a3.png

3、XSRF 攻击

  • 我向你发送一封电子邮件,邮件标题很吸引人
  • 但邮件正文隐藏着 <img src="xxx.com/pay?id=200" />
  • 你一查看邮件,就帮我购买了 id 是 200 的商品

4、XSRF 预防

  • 使用 post 接口
  • 增加验证,例如密码、短信验证码、指纹等
相关文章
|
6月前
|
并行计算 前端开发 JavaScript
Web Worker:让前端飞起来的隐形引擎
在现代 Web 开发中,前端性能优化是一个至关重要的课题,尤其是对于计算密集型的应用,如图像处理、视频处理、大规模数据分析等任务。单线程的 JavaScript 引擎常常成为性能瓶颈,导致应用变得迟缓。Web Worker,作为一种强大的技术,使得前端能够在后台进行并行计算,从而实现高效的任务处理,不影响主线程的运行和用户的交互体验。
559 108
|
6月前
|
JavaScript 前端开发 Java
前端框架选择之争:jQuery与Vue在现代Web开发中的真实地位-优雅草卓伊凡
前端框架选择之争:jQuery与Vue在现代Web开发中的真实地位-优雅草卓伊凡
654 72
前端框架选择之争:jQuery与Vue在现代Web开发中的真实地位-优雅草卓伊凡
|
8月前
|
移动开发 前端开发 JavaScript
前端web创建命令
本项目使用 Vite 搭建 Vue + TypeScript 开发环境,并基于 HTML5 Boilerplate 提供基础模板,快速启动现代前端开发。
117 2
|
8月前
|
Web App开发 编解码 移动开发
零基础音视频入门:你所不知道的Web前端音视频知识
本文回顾了Web端音视频的发展历程,同时还介绍了视频的编码、帧率、比特率等概念,提到了Canvas作为视频播放的替代方案,以及FFmpeg在音视频处理中的重要作用等知识。
246 1
|
8月前
|
存储 安全 算法
Java 集合面试题 PDF 下载及高频考点解析
本文围绕Java集合面试题展开,详细解析了集合框架的基本概念、常见集合类的特点与应用场景。内容涵盖`ArrayList`与`LinkedList`的区别、`HashSet`与`TreeSet`的对比、`HashMap`与`ConcurrentHashMap`的线程安全性分析等。通过技术方案与应用实例,帮助读者深入理解集合类的特性和使用场景,提升解决实际开发问题的能力。文末附带资源链接,供进一步学习参考。
200 4
|
8月前
|
存储 安全 Java
Java 集合面试题从数据结构到 HashMap 源码剖析详解及长尾考点梳理
本文深入解析Java集合框架,涵盖基础概念、常见集合类型及HashMap的底层数据结构与源码实现。从Collection、Map到Iterator接口,逐一剖析其特性与应用场景。重点解读HashMap在JDK1.7与1.8中的数据结构演变,包括数组+链表+红黑树优化,以及put方法和扩容机制的实现细节。结合订单管理与用户权限管理等实际案例,展示集合框架的应用价值,助你全面掌握相关知识,轻松应对面试与开发需求。
399 3
|
Java 程序员
Java社招面试中的高频考点:Callable、Future与FutureTask详解
大家好,我是小米。本文主要讲解Java多线程编程中的三个重要概念:Callable、Future和FutureTask。它们在实际开发中帮助我们更灵活、高效地处理多线程任务,尤其适合社招面试场景。通过 Callable 可以定义有返回值且可能抛出异常的任务;Future 用于获取任务结果并提供取消和检查状态的功能;FutureTask 则结合了两者的优势,既可执行任务又可获取结果。掌握这些知识不仅能提升你的编程能力,还能让你在面试中脱颖而出。文中结合实例详细介绍了这三个概念的使用方法及其区别与联系。希望对大家有所帮助!
640 60
|
12月前
|
前端开发
【2025优雅草开源计划进行中01】-针对web前端开发初学者使用-优雅草科技官网-纯静态页面html+css+JavaScript可直接下载使用-开源-首页为优雅草吴银满工程师原创-优雅草卓伊凡发布
【2025优雅草开源计划进行中01】-针对web前端开发初学者使用-优雅草科技官网-纯静态页面html+css+JavaScript可直接下载使用-开源-首页为优雅草吴银满工程师原创-优雅草卓伊凡发布
352 1
【2025优雅草开源计划进行中01】-针对web前端开发初学者使用-优雅草科技官网-纯静态页面html+css+JavaScript可直接下载使用-开源-首页为优雅草吴银满工程师原创-优雅草卓伊凡发布
|
前端开发 JavaScript 搜索推荐
HTML与CSS在Web组件化中的核心作用及前端技术趋势
本文探讨了HTML与CSS在Web组件化中的核心作用及前端技术趋势。从结构定义、语义化到样式封装与布局控制,两者不仅提升了代码复用率和可维护性,还通过响应式设计、动态样式等技术增强了用户体验。面对兼容性、代码复杂度等挑战,文章提出了相应的解决策略,强调了持续创新的重要性,旨在构建高效、灵活的Web应用。
337 6
|
JavaScript 前端开发 安全
2024年前端开发新趋势:TypeScript、Deno与性能优化
2024年前端开发迎来新趋势:TypeScript 5.0引入装饰器正式支持、const类型参数及枚举改进;Deno 1.42版推出JSR包注册表、增强Node.js兼容性并优化性能;性能优化策略涵盖代码分割、懒加载及现代构建工具的应用。这些变化推动前端开发向更高效率和安全性发展。