面试题手册

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

前端阅读 482024年6月24日 16:43

列举 3 种强制类型转换和 2 种隐式类型转换

强制类型转换强制类型转换 是指开发者显式地将一种数据类型转换为另一种数据类型。强制类型转换的例子:Number转换为String:使用 toString()方法来转换数字为字符串。例如:(123).toString(),结果为 "123"。String转换为Number:使用 Number()函数将字符串转换为数字。例如:Number("123"),结果为 123。非布尔值转换为布尔值:使用 Boolean()函数将非布尔值转换为布尔值。例如:Boolean(1),结果为 true。 隐式类型转换隐式类型转换,又被称为隐式类型强制转换,是指JavaScript引擎在处理表达式时自动完成的类型转换。隐式类型转换的两个例子:加法运算符:当通过加法运算符加入字符串和非字符串(数字,布尔值等)时,非字符串将被转换为字符串。例如:"5" + 3,结果为 "53"。相等性比较:如果比较的值具有不同的类型,JavaScript会尝试通过诸如转换字符串为数字,或转换布尔值为数字等方式,来进行比较。例如:"5" == 5,结果为 true。
前端阅读 342024年6月24日 16:43

Web前端安全攻击手段有哪些?以及应该如何做相应的防御措施?

Web前端安全攻击主要有以下几种常见的方式:XSS攻击(跨站脚本攻击):攻击者通过在目标网站上注入恶意的HTML代码,当用户浏览该网站时就会运行这些恶意代码。防御措施:输入验证与过滤、输出编码、使用CSP(内容安全策略)防止不安全的动态脚本执行。CSRF攻击(跨站请求伪造):攻击者诱导用户点击链接,使用用户的登录凭证发送恶意请求。防御措施:使用CSRF token,验证每个请求。点击劫持:攻击者将透明的恶意网站覆盖在真实网站上,诱导用户在不知情的情况下进行恶意操作。防御措施:使用X-FRAME-OPTIONS来防止网页被iframe调用。DoS攻击(拒绝服务攻击):恶意请求过多使得服务无法处理正常的请求。防御措施:限制访问频率、使用CDN(内容分发网络)等方式分散流量。SQL注入攻击:攻击者通过输入特殊的SQL查询语句,来获取数据库的敏感信息。防御措施:使用预处理语句(Prepared Statements)或参数化的SQL命令,拒绝直接执行动态生成的SQL语句。上传恶意文件:攻击者通过上传恶意文件,例如包含病毒或者后门的文件,来破坏服务器或者网页。防御措施:限制可上传文件类型,扫描上传的文件以防止上传恶意软件,对上传文件名严格过滤等。
前端阅读 1312024年6月24日 16:43

为什么有移动端 1px 的问题以及有什么方案可以解决移动端 1px 的问题?

为什么存在 1px 问题移动端的1px问题主要是由于设备的物理像素和逻辑像素的不同所致。在高清显示屏(即设备独立像素比例devicePixelRatio大于1)的设备上,一个CSS像素可能会对应多个设备像素。这就导致了CSS的1px边框在高清屏上显示得比预期更粗。 1px以下是一些主要的解决1px问题的方案:视口缩放(Viewport Scale)视口缩放的原理是将页面的视口设置为设备宽度的一半或一部分,然后布局以这个新的视口宽度为基准进行。由于视口被缩小,一个CSS像素也就对应了更少的设备像素。 <meta name="viewport" content="width=device-width,initial-scale=.5,maximum-scale=.5,user-scalable=no">使用Media Query可以使用CSS的Media Query配合devicePixelRatio设定不同dpr下的边框样式,这样可以确保在不同dpr的设备上边框都看起来只有1px。 @media only screen and (-webkit-device-pixel-ratio: 2) { .border { border-width: 0.5px; } }使用伪元素 + transform: scaleY(.5)/scaleX(.5)利用伪元素,将元素的边框做一次.5的缩放,使得高清屏下1px边框变细。 .border:after { content: ''; position: absolute; bottom: 0; background: #000; width: 100%; height: 1px; transform: scaleY(.5); -webkit-transform: scaleY(.5); }使用SVG可以使用SVG矢量图形来实现1px边框。由于SVG是矢量图形,在任何分辨率的屏幕上都会很清晰。 <svg width="100%" height="1" xmlns="http://www.w3.org/2000/svg" version="1.1"> <line x1="0" y1="0" x2="100%" y2="0" stroke="black" stroke-width=".5"/> </svg>边框图片(Border Image)border-image 属性可以让你使用图像作为边框,你可以自由定义图像如何填充和拉伸。以上只是其中的一些常用解决方案,每种方法都有其适用范围和局限。在实际开发过程中,需要根据具体的应用情况选择最合适的方案。
前端阅读 1392024年6月24日 16:43

CSS 选择器有哪些

CSS 选择器用于决定某个HTML元素上应用哪些样式。以下是一些常用的CSS选择器:1. 元素选择器基于HTML元素的名称选择元素。例如,选择所有的 <p> 元素:p { color: red;}2. id 选择器使用HTML元素的 id 属性选择特定元素。id 选择器在CSS中使用 #" 符号定义。例如,选择 id="intro"的元素:#intro { color: blue;}3. 类选择器使用HTML元素的 class 属性选择特定元素。类选择器在CSS中使用 . 符号定义。例如,选择 class="highlight"的所有元素:.highlight { background-color: yellow;}4. 属性选择器可以选择带有指定属性的HTML元素。例如,选择所有带有 target 属性的 <a> 元素:a[target] { background-color: pink;}5. 伪类选择器用于选择HTML元素的特定状态。例如,选择鼠标悬停在上面的 <a> 元素:a:hover { color: green;}6. 伪元素选择器用于选择元素的某一部分。例如,选择每个 <p> 元素的第一行:p::first-line { color: orange;}7. 组合选择器有时候,我们需要选择满足多项条件的元素,这时候就可以使用组合选择器。常见的组合选择器如下:后代选择器(空格)div p { color: brown;}子代选择器(>)div > p { color: purple;}相邻兄弟选择器(+)div + p { color: gray;}一般兄弟选择器(~)div ~ p { color: teal;}
前端阅读 792024年6月24日 16:43

display: none;与visibility: hidden;的区别

联系:它们都能让元素不可见区别:display:none;会让元素完全从渲染树中消失,渲染的时候不占据任何空间;visibility: hidden;不会让元素从渲染树消失,渲染时元素继续占据空间,只是内容不可见。display: none;是非继承属性,子孙节点消失由于元素从渲染树消失造成,通过修改子孙节点属性无法显示;visibility: hidden;是继承属性,子孙节点由于继承了 hidden 而消失,通过设置 visibility: visible,可以让子孙节点显示。修改常规流中元素的 display 通常会造成文档重排。修改 visibility 属性只会造成本元素的重绘。读屏器不会读取 display: none;元素内容;会读取 visibility: hidden;元素内容。
前端阅读 822024年6月24日 16:43

link与@import的区别

link和 @import都是用于在HTML文档中链接和导入CSS样式的方法,但它们之间存在一些关键区别:表现形式link是HTML标签,这种方式通过HTML的 <link>元素来提供链接;@import是CSS提供的方式,可以在一个CSS样式表中导入其他样式表;加载方式link在页面一开始加载时就会同时加载CSS文件;@import则是在页面加载完之后再加载CSS,因此如果CSS文件较多、较大时,可能会导致页面加载不同步,影响网页性能;兼容性link是XHTML标签,无论是早期的HTML版本还是现在的XHTML,都支持 link标签;@import是在CSS2.1才出现的,所以早期的浏览器不支持 @import方式;使用条件link可以定义RSS、链接到打印版的CSS等,更具有扩展性;@import只能加载CSS以上就是 link和 @import的主要区别。
前端阅读 682024年6月24日 16:43

Form 表单可以执行跨域请求吗?

HTML表单(form)是可以执行跨域请求的。在Web开发中,跨域请求指的是从一个源(domain、协议、端口)发起的请求试图获取另一个源上的资源。通常,出于安全考虑,浏览器会实施同源策略(Same-Origin Policy),这意味着如果使用XMLHttpRequest或Fetch API来发起Ajax请求,那么请求通常会受到限制,除非目标服务器明确允许跨源资源共享(CORS,Cross-Origin Resource Sharing)。但是,HTML表单不受同源策略的限制,因此可以向任何URL发起POST或GET请求,即使这个URL指向的是另一个域。当表单提交时,浏览器会将用户输入的数据作为请求参数发送到表单的 action属性所指定的URL。不过,使用表单进行跨域提交时,浏览器会导航到响应页面,也就是说用户的当前页面会被新页面替换。下面是一个简单的表单跨域请求的例子:<form action="https://example.com/api/submit" method="POST"> <input type="text" name="username" value="User"> <input type="text" name="password" value="Pass"> <input type="submit" value="Submit"></form>在此示例中,表单数据将提交给位于 example.com域的服务器,即使表单所在的HTML页面可能托管在不同的域上。当用户点击提交按钮时,浏览器会将表单数据作为POST请求的一部分发送到 example.com。需要注意的是,即使表单可以跨域提交,服务端仍然需要处理来自不同源的请求。此外,跨域表单提交不会提供Ajax那样的客户端JavaScript接口来访问响应内容,除非服务器在响应中包含适当的CORS头部信息。
前端阅读 422024年6月24日 16:43

什么是 web 语义化,有什么好处

什么是 Web 语义化Web 语义化是指通过 HTML 标记表示页面包含的信息,包含了 HTML 标签的语义化和 css 命名的语义化。HTML 标签的语义化通过使用包含语义的标签(如 h1-h6)恰当地表示文档结构 。CSS 命名的语义化 为 html 标签添加有意义的 class,id 补充未表达的语义,如通过添加符合规则的 class 描述信息。 Web 语义化的好处需要语义化的主要理由有以下几点:去掉样式后页面呈现清晰的结构盲人使用读屏器更好地阅读搜索引擎更好地理解页面,有利于收录便团队项目的可持续运作及维护
前端阅读 292024年6月24日 16:43

JavaScript 执行过程分为哪些阶段?

JavaScript 的执行过程大致可以分为以下几个阶段:1. 解析(Parsing)在这一阶段,JavaScript 引擎会读取源代码,并将其解析成抽象语法树(AST)。抽象语法树是一种深层次的、结构化的代码表达方式,能够以树形结构表现代码中的每个语句、表达式等元素。解析过程中,如果遇到语法错误,会抛出错误,停止进一步执行。2. 编译(Compilation)JavaScript 引擎(如V8)通常会将解析后的代码进行即时编译(JIT)。编译器会先生成字节码,这是一种低级的、比源代码更接近机器语言的代码。随后根据程序的执行情况,编译器可能会把热点代码(经常执行的代码)编译成优化的机器码,提高执行效率。3. 执行(Execution)编译后得到的字节码或机器码被送到 JavaScript 引擎的执行环境中执行。在执行过程中,会进入下面的子阶段:创建执行上下文(Execution Context):首先会创建全局执行上下文,随后每当调用一个函数时,就会为该函数创建一个新的执行上下文。执行上下文包括变量对象、作用域链和 this 引用等信息。变量提升(Hoisting):在执行代码前,函数声明和变量(声明)会被提升到它们各自的执行上下文的顶部。变量会初始化为 undefined,而函数则会完整地提升。执行代码(Running Code):按照执行上下文中的代码逐行执行,进行变量赋值、函数调用等。垃圾回收(Garbage Collection):在执行过程中,引擎会进行内存管理,自动释放那些不再被需要的内存空间。4. 优化(Optimization)在代码执行的过程中,某些代码可能会被执行多次,引擎会尝试对这些频繁执行的代码进行优化。例如,在V8引擎中,有一个称为“TurboFan”的优化编译器,它可以根据代码执行的特点对代码进行优化,提高性能。如果优化假设失败了(即出现了“去优化” deoptimization),引擎还可以将代码回退到一个较少优化的版本。5. 回收(Deoptimization & Garbage Collection)对于那些不再需要的数据和优化,JavaScript 引擎会进行去优化和垃圾回收,以保证内存的高效使用。例子:假设我们有这样一个简单的 JavaScript 函数:function sum(a, b) { return a + b;}let result = sum(5, 3);首先,该函数会被解析成 AST。然后,它可能会被编译成字节码,当我们调用 sum(5, 3) 时,会创建一个新的执行上下文,包含 a 和 b 的参数以及任何局部变量。在这个上下文中,a 和 b 被赋予了值 5 和 3,函数执行,并返回结果 8。这个过程中可能还包括了对 sum 函数的优化,如果函数被频繁调用。最后,当执行上下文离开作用域,如果没有其他引用指向其中的数据,垃圾回收器最终会清理掉这些对象。
前端阅读 432024年6月24日 16:43

JavaScript 为什么要区分微任务和宏任务?

JavaScript 区分微任务(microtasks)和宏任务(macrotasks)主要是为了有效地管理异步操作的执行时机和顺序。这两种类型的任务允许 JavaScript 引擎维持一个控制异步操作何时何地执行的精细化调度机制。宏任务 (Macrotasks)宏任务通常是浏览器的主要任务,包括但不限于:setTimeoutsetIntervalI/O 操作UI 渲染事件处理(如点击、滚动事件)每次执行栈为空时,事件循环都会从任务队列中取出一个宏任务执行。微任务 (Microtasks)微任务通常是需要快速响应的任务,执行时机在每个宏任务之后,以及JavaScript执行环境准备好以后。微任务包括:Promise 回调(例如.then、.catch和.finally)MutationObserver 的回调queueMicrotask函数执行顺序在每个宏任务执行完毕后,在执行下一个宏任务之前,JavaScript 引擎会处理所有队列中的微任务。这意味着微任务总是在当前宏任务结束和下一个宏任务开始之间执行,并在新的UI渲染前完成。这样可以确保异步操作的快速响应,同时因为微任务的延迟更小,所以适合高优先级的任务。为什么区分区分微任务和宏任务的原因包括:性能优化: 通过微任务,JavaScript 可以在不影响用户界面渲染的情况下,快速执行简单的操作,如承诺的解决。这提高了应用程序的响应速度和性能。控制异步操作顺序: 微任务和宏任务的区分允许开发者控制异步操作的执行顺序。例如,一个由Promise产生的微任务可以确保其在下次UI渲染之前解决,而setTimeout可能会推迟到下一个宏任务。避免阻塞: 对于需要快速执行的代码,使用微任务可以避免阻塞宏任务队列,这有助于避免长时间运行的任务阻塞UI更新。示例假设我们有以下代码:console.log('宏任务开始');setTimeout(() => { console.log('宏任务');}, 0);Promise.resolve().then(() => { console.log('微任务');});console.log('宏任务结束');执行顺序会是这样的:打印"宏任务开始"宏任务结束时,设置了一个setTimeout打印"宏任务结束"当前宏任务已结束,开始执行微任务队列中所有任务打印"微任务"微任务队列为空,开始下一个宏任务打印"宏任务"通过这个例子,我们可以看到微任务总是在当前执行栈清空后立即执行,而宏任务的执行则可能会因为任务队列中的其他宏任务而延迟。这种机制允许JavaScript有效地处理异步事件,同时保持对执行顺序的细粒度控制。
前端阅读 262024年6月24日 16:43

jsonp 为什么不支持 post 方法?

JSONP(JSON with Padding)是一种利用<script>标签不受同源策略限制的特性来实现跨源请求的技术。因为<script>标签的初衷是加载静态的JavaScript文件,所以<script>标签仅支持GET方法来请求资源,它并不支持POST方法。这就是为什么JSONP不支持POST方法的根本原因。当使用JSONP进行通信时,您会将请求参数包含在URL中,并通过动态创建<script>标签的方式将其发出。服务器接收到GET请求后,将数据包裹在一个函数调用中,并将其作为响应返回。客户端定义好回调函数后,这段包裹着JSON数据的JavaScript被执行,回调函数便会被调用并处理返回的数据。例如,假设您的页面需要从http://example.com获取一些用户数据,您可能会发送如下的JSONP请求:<script type="text/javascript"> // 定义回调函数 function handleResponse(data) { console.log('Received data:', data); }</script><script type="text/javascript" src="http://example.com/data?callback=handleResponse"></script>服务器端需要接收到callback参数后,把数据包装在该函数调用中:// 服务器端响应handleResponse({ "user": "Alice", "age": 25 });如上所述,JSONP请求的本质是一个GET请求,它是通过<script>标签的src属性来发起的。因此,它不能使用POST方法,后者通常用于传输大量数据或者发送需要安全传输的数据。如果您需要进行跨域的POST请求,可以考虑使用更现代的技术,如CORS(跨源资源共享),它允许在各种HTTP方法中使用跨源请求,同时提供了更好的安全性。
前端阅读 352024年6月24日 16:43

Ajax 如何处理请求跨域问题?CORS 如何设置?

Ajax 处理请求跨域问题Ajax(Asynchronous JavaScript and XML)本身是不支持跨源请求的,这是因为浏览器出于安全考虑实施的同源策略(Same-Origin Policy)。同源策略限制了来自不同源的文档或脚本对当前文档读取或设置某些属性的能力。不过,有几种方法可以绕过这个限制来实现跨源请求:JSONP(JSON with Padding):这是一种老的技巧,它利用 <script>标签不受同源策略限制的特点来进行跨域请求。服务器返回的响应拼接一个函数调用作为JavaScript代码。这种方法的缺点是它只能用于GET请求,并且存在一定的安全隐患。CORS(Cross-Origin Resource Sharing):这是现在推荐的方法,它允许服务器显式地指定哪些源可以访问该服务器上的资源。CORS是一个 W3C 标准,它通过在服务器上设置特定的HTTP头来工作。代理服务器:使用一个服务器充当中间人的角色,接受来自客户端的跨源请求,然后转发请求到目标服务器。代理服务器接收到响应后,再将数据返回给客户端。这种方法需要额外的服务器端配置。WebSockets: WebSockets提供了一个全双工的通信渠道,可以在浏览器和服务器之间建立持久连接。虽然WebSocket协议与HTTP不同,但它可以绕过同源策略,从而实现跨域通信。CORS 设置当你控制服务器时,可以通过设置HTTP响应头来启用CORS。这些头部主要包括:Access-Control-Allow-Origin: 指定了哪些网站可以参与跨域资源共享。例如,设置为 *代表允许所有域进行跨域请求,但出于安全考虑通常会指定明确的域名。Access-Control-Allow-Methods: 指定了允许的HTTP请求方法,如 GET, POST, PUT, DELETE, OPTIONS等。Access-Control-Allow-Headers: 在实际请求中允许自定义的HTTP头部字段列表。Access-Control-Allow-Credentials: 表示是否允许发送Cookie。如果服务器端设置了这个值为 true,那么前端请求的时候也必须将 withCredentials属性设置为 true。Access-Control-Expose-Headers: 允许客户端访问的服务器白名单头部字段。Access-Control-Max-Age: 表明在多少秒内,不需要再发送预检验请求(对于某一资源的预检请求结果的有效期)。下面是一个简单的例子,展示了如何在Node.js的Express框架中设置CORS响应头:const express = require('express');const app = express();app.use((req, res, next) => { res.header('Access-Control-Allow-Origin', 'https://example.com'); res.header('Access-Control-Allow-Methods', 'GET, POST, PUT, DELETE, OPTIONS'); res.header('Access-Control-Allow-Headers', 'Origin, X-Requested-With, Content-Type, Accept, Authorization'); res.header('Access-Control-Allow-Credentials', true); if (req.method === 'OPTIONS') { res.header('Access-Control-Max-Age', '86400'); return res.status(200).send(); } next();});// 你的其他路由和逻辑app.listen(3000, () => { console.log('Server running on port 3000');});
前端阅读 752024年6月24日 16:43

base64 为什么能提升性能?

Base64编码通常不是直接用来提升性能的,而是用来确保二进制数据可以通过仅支持文本格式的传输层安全地传输。Base64将二进制数据编码为ASCII字符串,这使得它可以在不支持二进制数据的系统(如电子邮件)中传输。虽然Base64编码的数据比原始二进制数据大约增加了33%,但在某些情况下,它可以间接地提升性能:减少HTTP请求:在Web开发中,Base64编码通常用于将小的图片或其他文件直接嵌入到HTML或CSS中。这样做的好处是可以减少浏览器发起的HTTP请求的数量,因为所有的资源都包含在了主文档中。少了额外的请求,网页加载时间就会缩短,间接提高了用户体验和性能。举个例子,如果一个网页中有多个小图标,通常的做法可能是每个图标一个HTTP请求来获取图像文件。如果将这些图标的图片用Base64编码后嵌入到CSS中,就可以将多个HTTP请求合并为一个请求,从而减少了服务器的请求负载和网络延迟,提高了页面的加载速度。数据URI方案:Base64编码可以使用数据URI方案在Web页面中直接嵌入图像数据,这样可以避免服务器配置对小文件的较慢响应时间。服务器对小文件的处理往往不如大文件高效,因为涉及到磁盘I/O等开销。通过避免这些小文件请求,可以在服务器端节省资源,从而提升性能。安全和兼容性:有些系统不支持二进制数据的传输,或者在传输过程中可能会因为某些字符(如NUL byte)的存在而出现问题。在这种情况下,Base64编码提供了一种可靠的方法来处理和传输数据,避免了潜在的数据损坏和传输错误,从而确保系统的顺畅运行和性能。总之,Base64编码本身增加了数据量,理论上会降低传输效率,但通过减少HTTP请求的数量、充分利用缓存、避免小文件请求开销以及提高数据的安全性和兼容性,它可以在特定场景下间接提升系统的整体性能。
前端阅读 392024年6月24日 16:43

JS 如何去除 url 中的#号?

如果我们要在JavaScript中去除URL中的 # 号以及后面的部分,我们可以使用 window.location 对象,具体是 window.location.href 属性,再结合 String 对象的 split 方法。请看以下例子: // 假设当前URL为: https://www.example.com/page.html#section1// 使用 JavaScript 获取当前URL并去除 # 及之后的部分function removeHashFromUrl() { var currentUrl = window.location.href; var urlWithoutHash = currentUrl.split('#')[0]; window.location.href = urlWithoutHash; // 如果需要导航到去除hash的URL return urlWithoutHash; // 如果只是需要获取新的URL而不导航}var newUrl = removeHashFromUrl();console.log(newUrl); // 输出:https://www.example.com/page.html如果我们是在后端处理URL字符串,比如在Node.js环境或者其他不涉及浏览器的上下文中,我们可以简单地使用字符串处理方法。这里是用Node.js中的JavaScript例子:// 假设有一个URL字符串var url = "https://www.example.com/page.html#section1";// 去除URL中的 # 及之后的部分function removeHashFromUrl(url) { return url.split('#')[0];}var newUrl = removeHashFromUrl(url);console.log(newUrl); // 输出:https://www.example.com/page.html在Python中处理URL也很简单,我们可以使用内置的 urlparse库,这样可以更加优雅地处理复杂的URL。这是一个Python例子:from urllib.parse import urlparse# 假设有一个URL字符串url = "https://www.example.com/page.html#section1"# 去除URL中的 # 及之后的部分parsed_url = urlparse(url)new_url = parsed_url.scheme + "://" + parsed_url.netloc + parsed_url.pathprint(new_url) # 输出:https://www.example.com/page.html以上提供了去除URL中 # 号的几种方法,具体使用哪种取决于具体的应用场景以及开发环境。在前端JavaScript开发中,我们通常可能会涉及到浏览器的 window.location 对象,而在服务器端或者其他一些脚本处理中,则可能会使用字符串处理函数或者URL解析库。
前端阅读 372024年6月24日 16:43

addEventListener 的第三个参数的作用是什么?

addEventListener 方法是 JavaScript 中常用来为元素添加事件监听器的方法。这个方法可以让开发者指定当某个事件在目标元素上触发时,应该调用的回调函数。addEventListener 方法通常接收三个参数:type: 字符串,表示监听事件类型的名称,比如 click, mouseover 等。listener: 函数,事件触发时浏览器调用的函数。options or useCapture: (可选)布尔值或者是一个对象。这是第三个参数,它指定了事件处理的更多选项。当第三个参数是布尔值时,它指的是 useCapture。如果 useCapture 设置为 true,则表示在捕获阶段触发事件处理函数;如果设置为 false 或者省略,则表示在冒泡阶段触发事件处理函数。在 DOM 事件处理中,事件传播分为三个阶段:捕获阶段、目标阶段和冒泡阶段。默认情况下,事件监听器只在冒泡阶段被调用。如果第三个参数是一个对象,它可以包含多个属性,如下所示:capture: 布尔值,和直接提供 useCapture 作为布尔值的效果一样。once: 布尔值,如果为 true,监听器会在添加之后第一次触发时自动移除。passive: 布尔值,如果为 true,表明监听器永远不会调用 preventDefault()。如果监听器确实调用了这个函数,客户端将会忽略它并且可能给出一个警告。例如,如果我们想要在用户第一次点击按钮时做出反应,并且希望在捕获阶段而不是冒泡阶段处理事件,我们可以这样写代码:const button = document.querySelector('#myButton');button.addEventListener('click', (event) => { // 处理点击事件 console.log('Button clicked!');}, { capture: true, once: true });在这个例子中,{ capture: true, once: true } 作为第三个参数传递,确保了监听器在捕获阶段执行,并且只执行一次。
前端阅读 292024年6月24日 16:43

cros 的简单请求和复杂请求的区别是什么?

CORS,即跨源资源共享(Cross-Origin Resource Sharing),是一种允许在一个源(origin)上的网页获取访问另一个源上资源的机制。它是一种安全特性,可以让网站的前端代码安全地进行跨域请求,而不会暴露用户数据。CORS 请求分为两类:简单请求(simple requests)和复杂请求(preflighted requests)。它们之间的区别主要体现在请求的方式和所发送内容上。简单请求简单请求满足以下条件:请求方法是以下三种方法之一:GETPOSTHEADHTTP的头信息不超出以下几种字段:AcceptAccept-LanguageContent-LanguageLast-Event-IDContent-Type(只限于三个值application/x-www-form-urlencoded、multipart/form-data、text/plain)简单请求的例子:GET /some/resource HTTP/1.1Host: api.example.comAccept-Language: en-USContent-Type: text/plain当浏览器判断一个请求为简单请求时,它会直接发起跨域请求,并在请求中携带Origin头部信息。服务端会检查这个Origin,决定是否允许这个跨域请求。复杂请求复杂请求通常指不满足以上简单请求条件的所有其他请求,例如:使用了 PUT、DELETE、CONNECT、OPTIONS、TRACE、PATCH 等 HTTP 方法。发送的 HTTP 头部信息超出了简单请求允许的范围。Content-Type 的值不属于简单请求中允许的三个值。在发送复杂请求之前,浏览器会先发起一个 OPTIONS 请求,这被称为“预检”请求(preflight request),用来确认真正的请求是否安全可被服务器接受。预检请求的例子:OPTIONS /data/resource HTTP/1.1Host: api.example.comOrigin: http://example.comAccess-Control-Request-Method: POSTAccess-Control-Request-Headers: X-Custom-Header如果服务器允许这样的请求,它会在响应的 HTTP 头部中包含Access-Control-Allow-Origin、Access-Control-Allow-Methods和Access-Control-Allow-Headers等字段,明确告知客户端是否可以进行实际的请求。之后,浏览器才会发送实际的请求。总结简单来说,简单请求是对CORS更宽容的请求,直接发起并通过Origin头部判断是否允许跨域;而复杂请求需要先进行一次额外的预检通信以确认安全性,只有在预检通过后,实际的请求才会发起。这个机制确保了敏感操作(如对数据的修改)在跨域时能够得到恰当的安全检查。
前端阅读 292024年6月24日 16:43

JavaScript 的遍历方法中,在 map 和 for 中调用异步函数的区别是什么?

在JavaScript中,map和for循环是遍历数组的两种常见方法,但在处理异步函数时,它们的行为有显著差异。使用map调用异步函数map函数是Array原型上的一个方法,它对数组中的每个元素执行一个由你提供的函数,并返回一个新的数组,该数组是由原数组中每个元素调用处理函数得到的结果组成的。当你在map内使用异步函数时,每次迭代都会立即发起异步操作,但不会等待上一个完成,这意味着所有异步操作几乎是同时发起的。map不会等待异步函数的解决,它会立即继续到下一次迭代。例如,如果你使用map遍历数组,并在每个元素上调用一个返回Promise的异步函数:let promises = [1, 2, 3].map(async (num) => { let result = await someAsyncFunction(num); return result;});这里,promises数组将包含三个Promise对象,这些Promise对象是someAsyncFunction返回的,并且他们将并行执行。使用for循环调用异步函数使用传统的for循环,你可以更容易地控制异步函数的执行顺序。如果在循环内部使用await,你可以确保每次迭代都等待上一个异步操作完成再继续。例如,使用for循环顺序执行异步操作:let results = [];for (let num of [1, 2, 3]) { let result = await someAsyncFunction(num); results.push(result);}在这段代码中,someAsyncFunction会为数组中的每个元素顺序执行。第二次迭代会等待第一次迭代中的异步操作完成,以此类推。这意味着异步操作是串行执行的。总结使用map调用异步函数时,所有异步操作几乎同时开始,它们是并行的,最后你得到一个Promise对象的数组。使用for循环(或其他类型的循环,如for...of、for...in、while等)并结合await调用异步函数时,操作将按顺序一个接一个地执行,即串行执行。因此,选择哪种方法取决于你是否需要并行或串行执行异步操作。如果操作之间没有依赖,并且你想最大限度地提高效率,可以使用map。如果操作必须按照一定的顺序执行,或者一个操作的输出是另一个操作的输入,那么使用for循环会更合适。
前端阅读 222024年6月24日 16:43

for..of 和 for...in 是否可以直接遍历对象?

for...of 循环是在ES6中引入的,它专门用于遍历可迭代对象的元素,如数组、字符串、Map、Set 等这些实现了迭代器接口的数据结构。所谓的可迭代对象就是那些具有 Symbol.iterator 属性的对象。例如,数组是可迭代对象,可以使用 for...of 遍历其元素:let array = [10, 20, 30];for (let value of array) { console.log(value);}// 输出:// 10// 20// 30然而,普通对象不是可迭代的,没有实现 Symbol.iterator 方法,因此不能直接使用 for...of 遍历它的属性。尝试使用 for...of 直接遍历一个对象会导致一个错误:let obj = {a: 1, b: 2, c: 3};for (let value of obj) { console.log(value);}// TypeError: obj is not iterable另一方面,for...in 循环是用来遍历一个对象的所有可枚举属性的键,包括继承的可枚举属性。它不仅可以遍历普通对象的属性,还可以遍历数组(虽然通常不推荐这样做,因为它会返回数组索引,而且可能会遍历到原型链上的属性)。使用 for...in 遍历对象的例子:let obj = {a: 1, b: 2, c: 3};for (let key in obj) { console.log(key + ': ' + obj[key]);}// 输出:// a: 1// b: 2// c: 3总结一下,for...of 用于遍历可迭代对象的元素,而 for...in 用于遍历对象的所有可枚举属性的键。因此,for...of 不能直接遍历普通对象,而 for...in 可以。
前端阅读 92024年6月24日 16:43

分别介绍事件冒泡、事件代理、事件捕获,以及它们的关系?

事件冒泡 (Event Bubbling)事件冒泡是一种事件传播机制,在这种机制下,当一个元素上的事件被触发时,这个事件会从触发元素开始,逐级向上传播至最外层的父元素。这种传播方式允许在父元素上监听并处理来自子元素的事件。事件冒泡通常用于减少事件处理器的数量,并且简化事件管理。例子: 假设我们有一个按钮(<button>)位于一个段落(<p>)元素内,该段落又位于一个容器(<div>)元素内。如果用户点击了按钮,那么点击事件会首先在按钮元素上触发,然后依次向上冒泡至段落元素,最终到达容器元素。事件代理 (Event Delegation)事件代理是一种借助事件冒泡机制实现的事件处理模式。它通过在父元素上设置一个事件监听器来管理所有子元素的同类事件。这样可以避免在每个子元素上单独设置事件监听器,从而提高效率和性能,尤其是在动态添加或删除子元素的情况下。例子: 假如我们有一个任务列表,列表中的每一项任务都需要一个点击事件监听器。使用事件代理,我们可以在任务列表的容器元素上设置一个点击事件监听器,而不是在每个任务项上单独设置。当点击事件发生并冒泡到容器元素时,我们可以检查事件的目标元素(event.target)来确定是哪个任务项被点击,并进行相应的处理。事件捕获 (Event Capturing)事件捕获是DOM事件流的另一部分,与事件冒泡相对应。在事件捕获阶段,事件从最外层的父元素开始传递,一直向下直到触发元素。主要的区别在于事件的传播方向:事件捕获是从外到内,而事件冒泡是从内到外。例子: 再次使用上面的按钮、段落和容器元素的场景,当用户点击按钮时,在事件捕获阶段,点击事件会首先到达最外层的容器元素,然后到达段落元素,最后到达按钮元素。它们的关系事件捕获和事件冒泡是DOM事件流的两个阶段。在实际的事件处理中,浏览器首先经过捕获阶段,从最外层的父元素向下传递到目标元素,然后是目标元素上的事件处理,接着是冒泡阶段,事件从目标元素开始向上逐级传播。事件代理则是利用了事件冒泡原理来简化事件管理。通过在父元素上监听事件,可以管理所有子元素的事件,而无需在每个子元素上单独绑定事件监听器,使得代码更加简洁高效。在使用 addEventListener 方法时,我们可以指定第三个参数为 true 或 false 来明确选择是在捕获阶段还是冒泡阶段处理事件,默认值为 false,即在冒泡阶段处理。总之,事件捕获和事件冒泡共同构成了事件传播的完整过程,而事件代理则是一种利用这种传播机制的高效事件处理策略。