前端面试题手册

梳理高频技术问题,帮助你按主题复习和查漏补缺。

前端阅读 345月27日 01:10

Chrome 打开一个页面需要启动多少进程?分别有哪些进程?

Chrome 是多进程架构,主要进程:Browser 进程(1 个):主进程,管 UI(地址栏、书签)、网络请求、文件访问。协调其他进程。Renderer 进程(每个标签页一个):渲染页面内容(HTML、CSS、JS)。沙箱隔离,一个页面崩不影响其他页面。GPU 进程(1 个):处理 GPU 任务(CSS 3D 变换、WebGL、视频解码)。Plugin 进程(按需):隔离运行的插件(Flash 等,现已很少)。Network 进程(1 个,较新版本):独立的网络进程。Utility 进程(按需):音频、扩展等。同一站点(same-site)的标签页可能共享 Renderer 进程以节省内存。具体数量取决于你开了多少标签页、多少扩展。追问为什么 Chrome 用多进程而不是多线程?稳定性和安全性。如果一个标签页崩溃或卡死,只影响那个 Renderer 进程,不会拖垮整个浏览器。沙箱机制也只限制单进程的权限。缺点是内存占用高,所以 Chrome 引入了进程共享(同一站点共享 Renderer 进程)。Renderer 进程里是什么?Renderer 进程包含:主线程(JS 执行、样式计算、布局)、合成线程(处理滚动、动画)、Raster 线程(将绘制命令转成位图)、Worker 线程等。主线程阻塞是页面卡顿的主要原因。内存不够时 Chrome 会怎么做?Chrome 会动态管理:冻结后台标签页的进程(节省内存)、优先杀掉非活动标签页的 Renderer 进程、同一站点合用一个 Renderer 进程。开大量标签页时内存压力还是很大。
前端阅读 275月27日 01:09

浏览器为什么要有同源限制?

同源策略是浏览器最核心的安全机制——限制一个源(协议+域名+端口都相同)的脚本去访问另一个源的资源。没有同源限制会怎样?你打开了 bank.com 登录了,另一个标签页打开了 evil.com。evil.com 的 JS 可以直接用 fetch 发请求到 bank.com/api/transfer——浏览器会自动带上你的登录 cookie,钱就转走了。同源策略阻止了这个。同源限制主要限制:1) AJAX 跨域请求;2) 跨 iframe/窗口的 DOM 操作;3) localStorage/Cookie 的跨域访问。跨域解决方案:CORS(服务端设 Access-Control-Allow-Origin)、JSONP(老方法,利用 script 标签不受同源限制)、代理(同源服务端转发)。追问为什么 script/img 标签不受同源限制?这是历史原因和功能需求——网页需要加载 CDN 上的脚本和图片。但 <script> 加载完会以当前域执行(所以有 XSS 风险),<img> 只能渲染不能读取像素数据(防止窃取)。资源加载的宽松和 API 访问的严格是两个层面的事。同源策略能防止 CSRF 吗?不完全能。CSRF 的请求能发出去(form 提交、img 加载都可以跨域),同源策略只阻止 JS 读响应。防止 CSRF 得靠 CSRF Token 或 SameSite Cookie。跨域请求发出去但响应被拦截,服务端收到请求了吗?收到了。浏览器是收到响应后发现没有 CORS 头才拦截的。所以服务端需要用 CORS 头来控制,不能用"浏览器会拦截"来保护服务端。
前端阅读 1325月27日 01:06

Vue 3 如何在 setup 方法中获取组件实例?

Vue 3 Composition API 中,setup 里没有 this。要获取组件实例相关的能力,用对应的 API 而不是拿整个实例:获取 DOM 元素:const el = ref(null) + ref="el" 绑定,通过 el.value 访问获取路由实例:const router = useRouter(); const route = useRoute()获取 store:const store = useStore()(Pinia 或 Vuex 4)需要组件内部数据:直接在 setup 里定义 ref/reactive 并 return,模板可直接访问需要父组件数据:defineProps + defineEmits如果非拿实例不可(极少场景),Vue 3 提供了 getCurrentInstance():import { getCurrentInstance } from 'vue';const instance = getCurrentInstance();// instance.proxy 近似 Vue 2 的 this但官方不推荐依赖它,内部实现在版本间可能变化。追问Vue 3 为什么不给 this?Composition API 设计目标是函数式、可复用、类型推导友好。this 会绑定上下文,函数提取出去 this 就丢了。而 ref/reactive 是普通的 JS 值/对象,可以自由传递不丢失响应性。template ref 和 document.getElementById 怎么选?template ref 是 Vue 的方式——能保证在组件挂载后拿到元素,且不同组件实例互不干扰。document.getElementById 能拿到但可能拿到错误组件的元素(同页面多实例时),也不符合 Vue 的数据驱动理念。getCurrentInstance 在 setup 外能用吗?不能在 setup 外调用(如生命周期钩子的回调外)。它只在 setup 同步执行期间有效,异步回调里拿到的可能是 null。
前端阅读 725月27日 01:06

什么是双向绑定?Vue 是如何实现双向绑定功能的?

双向绑定 = 数据变化自动更新视图 + 用户输入自动更新数据。最典型的场景就是 <input v-model="msg">——你修改 input 的值,msg 跟着变;JS 里改 msg,input 显示跟着变。Vue 2 用 Object.defineProperty 劫持对象属性的 getter/setter 实现数据 → 视图的响应式,用 v-model 监听 input 事件实现视图 → 数据的反向同步。Vue 3 换成了 Proxy,能拦截更多操作(属性增删、数组索引),不需要像 Vue 2 那样用 $set 手动处理新增属性。追问v-model 的本质是什么?v-model="msg" 是语法糖,等价于 :value="msg" + @input="msg = $event.target.value"。对于 checkbox 是 :checked + @change,对于 select 是 :value + @change。双向绑定和单向数据流矛盾吗?不矛盾。Vue 仍然是单向数据流(父 → 子通过 props,子 → 父通过 emit)。v-model 只是父子通信的一种语法糖,本质上还是 prop + event 的模式。什么场景不适合用双向绑定?需要在赋值前做复杂转换的(用 computed + emit 更清晰)多个输入源同时修改一个数据的(容易互相覆盖)需要中间确认操作的(如提交时才正式更新)
前端阅读 145月27日 01:06

什么是深拷贝?什么是浅拷贝?

浅拷贝只复制对象的第一层属性,如果属性值是引用类型(对象、数组),拷贝的是引用地址——修改拷贝会影响原对象。深拷贝递归复制所有层级,新旧对象完全独立。// 浅拷贝const obj = { a: 1, b: { c: 2 } };const shallow = { ...obj }; // 或 Object.assign({}, obj)shallow.b.c = 3;console.log(obj.b.c); // 3 — 原对象被影响!// 深拷贝const deep = JSON.parse(JSON.stringify(obj)); // 最简单的方式... 展开和 Object.assign 都只做浅拷贝。JSON.parse(JSON.stringify(...)) 能做深拷贝但有局限:无法处理 undefined、函数、Date、RegExp、循环引用。追问JSON 方式深拷贝有哪些坑?undefined、函数、Symbol 直接被丢弃NaN 和 Infinity 变成 nullDate 变成字符串RegExp 变成空对象 {}循环引用直接报错原型链丢失生产环境怎么实现深拷贝?structuredClone()(浏览器原生 API,支持循环引用、Date、Map、Set 等,但不支持函数和 DOM 节点)lodash 的 cloneDeep自己实现递归拷贝 + WeakMap 处理循环引用(面试手写题常考这个)手写深拷贝时循环引用怎么处理?用 WeakMap 缓存已拷贝过的对象。每次拷贝前检查是否已经拷贝过,如果拷贝过直接返回缓存值。
前端阅读 675月27日 01:05

localStorage 对象有哪些 API?

localStorage 是浏览器提供的持久化键值存储,数据不随页面关闭而清除。API 一共就 5 个方法和 1 个属性:setItem(key, value):存数据。key 和 value 都是字符串。存对象需要用 JSON.stringifygetItem(key):取数据。key 不存在返回 nullremoveItem(key):删单个 keyclear():清空所有数据(注意:清的是当前域名下的所有 localStorage)key(index):按索引获取 key 名,用于遍历length:只读属性,返回存储的条目数存储对象的正确姿势:localStorage.setItem('user', JSON.stringify({ name: 'John', age: 30 }));const user = JSON.parse(localStorage.getItem('user'));追问localStorage 和 sessionStorage 有什么区别?生命周期:localStorage 永久存储,sessionStorage 关闭标签页就清除作用域:两者都是按域名隔离,但 sessionStorage 额外按标签页隔离API 完全一致localStorage 有什么限制?容量 5-10MB(不同浏览器不同)只能存字符串同步 API,大容量读写会阻塞主线程不能存敏感信息(明文存储,XSS 可直接读取)不支持过期时间,需要自己实现Cookie 和 localStorage 怎么选?需要随 HTTP 请求自动发送(如身份认证)→ Cookie纯前端存储、不随请求发送 → localStorage需要服务端可读 → Cookie(设置 HttpOnly 防止 XSS 读取)容量超过 4KB → localStorage(Cookie 限制 4KB)
前端阅读 265月27日 01:05

当添加原生事件不移除时,为什么会出现内存泄露?

核心原因:事件监听器持有 DOM 元素的引用,导致 DOM 元素被移除后无法被 GC 回收。具体链路:button.addEventListener('click', handler) 时,浏览器内部建立了 DOM元素 → handler 的引用关系你用 button.remove() 从 DOM 树移除按钮,但事件系统仍然保留着 handler 对 button 的引用如果 handler 又是闭包,引用着外层变量 → 整个闭包链上的变量都无法 GC多次重复后,内存中堆满了"已从 DOM 移出但无法回收的元素和函数",这就是内存泄漏解决办法:移除 DOM 元素前调用 removeEventListener。button.removeEventListener('click', handler);button.remove();或者在不需要关心个别元素时,用事件委托——把事件绑在父元素上。追问现代浏览器还会因为不移除事件导致泄漏吗?IE 时代这个问题最严重(IE 的 JS 引擎和 DOM 使用不同的 GC,循环引用是死穴)。现代浏览器的标记清除 GC 大部分循环引用能处理,但事件监听器 + 闭包的组合仍然会让本应被回收的对象变成"可达"——技术上不是传统意义的泄漏,但效果一样:内存释放不了。事件委托能解决这个问题吗?能。事件委托只绑一个监听器在父元素上,子元素增删都不需要管理事件。而且性能更好(减少监听器数量)。唯一注意:focus、blur、mouseenter、mouseleave 这些不冒泡的事件不能用委托。怎么在 DevTools 中排查内存泄漏?Performance 面板录一段操作,看 JS Heap 是否持续上升不回落 → Memory 面板取 Heap Snapshot,对比操作前后,看 Detached DOM 节点数量和 @ 引用关系。
前端阅读 345月27日 01:05

JavaScript 定时器为什么是不精确的?

setTimeout 和 setInterval 的执行时间不是精确的,根本原因就一条:JavaScript 是单线程的,定时器回调必须等主线程空闲才能执行。比如你设置 setTimeout(fn, 10),但此时主线程有个耗时 100ms 的计算任务在执行——fn 最早也要等 100ms 后才能跑,10ms 只是"最早执行时间",不是"准时执行时间"。其他影响因素:浏览器最小延迟:嵌套 5 次以上的 setTimeout 强制最小 4ms 间隔,后台标签页可能被降低到 1000ms事件循环优先级:微任务(Promise)比宏任务(定时器)优先执行GC 暂停:垃圾回收会短暂阻塞主线程追问需要精确计时怎么办?短时间精度:用 performance.now() 获取微秒级时间戳,自己在 requestAnimationFrame 里判断时间长时间精度:用 Web Worker(独立线程,不会被主线程阻塞),Worker 里用 setTimeout 相对更准音频场景:用 AudioContext.currentTime,这个不受主线程影响requestAnimationFrame 比 setInterval 更精确吗?rAF 本身也不是计时工具——它是"在浏览器下一次重绘前执行",频率取决于刷新率(60Hz 约 16.67ms)。它的优势是自动与渲染同步,不会做无效的更新。但作为定时器,它也有偏差。
前端阅读 225月27日 01:05

什么是 defineProperty 方法?什么时候需要用到它?

Object.defineProperty(obj, prop, descriptor) 让你精确控制对象属性的行为——是否可写、可枚举、可配置,还能定义 getter/setter。const obj = {};Object.defineProperty(obj, 'name', { value: '张三', writable: false, // 不可修改 enumerable: true, // for...in 可遍历 configurable: false // 不可删除、不可再次配置});核心价值在于拦截属性的读写。Vue 2 的响应式系统就是用 Object.defineProperty 劫持对象属性的 getter/setter,才能在数据变化时通知视图更新。追问defineProperty 和直接赋值有什么区别?直接赋值 obj.x = 1 创建的属性,writable、enumerable、configurable 默认都是 true,get/set 是 undefined。defineProperty 创建的属性,不显式指定的描述符默认都是 false。Vue 3 为什么换成 Proxy?defineProperty 有硬伤:无法检测属性的添加/删除(所以 Vue 2 需要 $set、$delete),对数组索引和 length 的拦截有坑。Proxy 能拦截 13 种操作,包括属性增删、in、delete、apply,不需要遍历对象属性递归劫持。什么场景下需要把属性设成不可枚举?给原型添加方法时(不想污染 for...in)框架内部属性(如 Vue 的 __ob__)任何不希望被 JSON.stringify 序列化或被 Object.keys 返回的属性
前端阅读 395月27日 01:05

什么是 CSRF 攻击?如何防御?

CSRF(跨站请求伪造)的攻击逻辑很简单:利用用户在其他网站已登录的身份,伪造请求执行非授权操作。典型场景:你登录了银行网站,cookie 还在。这时你打开了攻击者的网页,里面有一张隐藏的图片 <img src="https://bank.com/transfer?to=attacker&amount=10000">。浏览器加载这张"图片"时,自动带上了你的银行 cookie,银行收到请求以为是你的操作,转账就发生了。关键在于:浏览器发跨站请求会自动带上目标域名的 cookie——这是浏览器的默认行为。防御方式:Anti-CSRF Token:服务端生成随机 token,表单提交时带上,服务端验证。攻击者无法获取这个 tokenSameSite Cookie:Set-Cookie: SameSite=Strict,告诉浏览器跨站请求不要带这个 cookie。这是现代浏览器的标配方案Referer/Origin 检查:服务端验证请求来源,合法的请求才处理二次验证:关键操作(转账、改密码)要求重新输入密码或验证码追问CSRF 和 XSS 有什么区别?XSS 是在用户浏览器里执行恶意 JS(利用了网站对输入的信任),CSRF 是伪造用户的请求(利用了网站对浏览器身份的信任)。一句话:XSS 是"偷你的权限",CSRF 是"借用你的身份"。SameSite 三个值分别什么意思?Strict:任何跨站请求都不带 cookie,最安全。但用户从外部链接点进来是未登录状态Lax:大部分跨站请求不带 cookie,但"安全"的 GET 请求(如 <a> 链接点击)会带。兼顾安全和体验None:和以前一样,跨站请求都带 cookie,但必须同时设置 Secure(仅 HTTPS)Anti-CSRF Token 怎么防止攻击者自己获取一个 token?token 绑定到用户 session,攻击者拿到的 token 是绑给他自己的 session 的,放到跨站请求里服务端一对不上就拒了。关键不是"token 不能被获取",而是"token 和用户 session 绑定"。