JavaScript 面试经验(2024)

js事件循环机制

JavaScript 的事件循环(Event Loop)机制是其非阻塞单线程模型的核心。JavaScript 在浏览器环境中是单线程的,意味着它一次只能执行一个任务。但是,JavaScript 需要处理一些异步操作,如用户交互、网络请求等,这些操作可能需要等待一段时间才能完成。为了在不阻塞线程的情况下处理这些异步操作,JavaScript 使用了事件循环和消息队列。

以下是 JavaScript 事件循环机制的基本工作原理:

  1. 调用栈(Call Stack):JavaScript 引擎有一个调用栈,用于存储函数调用。当一个函数被调用时,它会被推入调用栈,当函数执行完成后,它会被弹出调用栈。
  2. 消息队列(Message Queue):也称为任务队列(Task Queue)或事件队列(Event Queue)。当异步操作(如 setTimeout、Promise、用户交互等)完成时,它们的结果(通常是一个回调函数)会被添加到消息队列中等待处理。
  3. 事件循环(Event Loop):事件循环会不断地检查调用栈是否为空。如果调用栈为空(即当前没有正在执行的代码),事件循环就会从消息队列中取出一个任务并将其推入调用栈中执行。这个过程会不断重复,形成一个循环。
  4. 微任务队列(Microtask Queue):除了普通的消息队列之外,JavaScript 还有一个微任务队列。Promise 的回调会被添加到微任务队列中。在每次事件循环的末尾,JavaScript 引擎会先检查微任务队列,如果有微任务,就会先执行微任务队列中的所有任务,然后再从消息队列中取下一个任务。这种处理方式是微任务比宏任务(消息队列中的任务)有更高的优先级。
  5. 浏览器渲染:在事件循环的间隙,浏览器会进行渲染操作。如果页面上有任何变化(如修改 DOM、CSS 等),浏览器会重新渲染页面。
  6. 异步操作:当 JavaScript 引擎遇到异步操作时(如 setTimeout、网络请求等),它会将操作放入后台执行,并立即返回执行结果(通常是一个 Promise 对象或 undefined)。当异步操作完成后,它的回调函数会被添加到消息队列或微任务队列中等待执行。

总结起来,JavaScript 的事件循环机制使得它能够非阻塞地处理异步操作,并保持应用程序的响应性。同时,通过消息队列和微任务队列,JavaScript 能够按照一定的优先级处理不同的异步任务。

css 元素水平垂直居中

在CSS中,有多种方法可以实现元素的水平垂直居中。以下是几种常用的方法:

1. 使用Flexbox(推荐)

Flexbox是一个现代的布局模式,可以轻松实现元素的水平和垂直居中。

css复制代码

.container {

display: flex;

justify-content: center; /* 水平居中 */

align-items: center; /* 垂直居中 */

height: 100vh; /* 假设你想要在视口高度内居中 */

}

.content {

/* 你的内容样式 */

}

2. 使用Grid

CSS Grid也可以很容易地实现元素的水平和垂直居中。

css复制代码

.container {

display: grid;

justify-items: center; /* 水平居中 */

align-items: center; /* 垂直居中 */

height: 100vh; /* 假设你想要在视口高度内居中 */

}

.content {

/* 你的内容样式 */

}

3. 使用定位(Positioning)和转换(Transform)

当你知道元素的尺寸时,可以使用定位和转换来实现居中。

css复制代码

.container {

position: relative;

height: 100vh; /* 假设你想要在视口高度内居中 */

}

.content {

position: absolute;

top: 50%;

left: 50%;

transform: translate(-50%, -50%); /* 水平和垂直移动元素自身尺寸的50% */

/* 你的内容样式 */

}

4. 使用CSS表格布局(不推荐,但兼容性好)

虽然这种方法在现代布局中不太常用,但它在旧版浏览器中仍然有效。

css复制代码

.container {

display: table-cell;

vertical-align: middle;

text-align: center; /* 水平居中,但仅对行内元素或文本有效 */

height: 100vh; /* 假设你想要在视口高度内居中 */

width: 某些值; /* 你需要设置一个宽度 */

}

.content {

display: inline-block; /* 或者其他行内元素类型 */

/* 你的内容样式 */

}

5. 使用行高(Line Height,仅适用于单行文本)

对于单行文本,你可以使用行高来实现垂直居中。

css复制代码

.container {

height: 某个值;

line-height: 与height相同的值; /* 使文本垂直居中 */

text-align: center; /* 水平居中 */

}

选择哪种方法取决于你的具体需求和目标浏览器的兼容性。在现代项目中,Flexbox和Grid通常是首选方法,因为它们提供了更大的灵活性和更简单的语法。

Promise

在JavaScript中,Promise是一种处理异步操作的机制,它是ES6(ECMAScript 2015)引入的一种语言特性。Promise可以看作是对异步操作的封装,它代表了一个从未完成到已完成(或失败)的操作,并可以返回操作的结果或错误。

Promise有三种状态:

  1. Pending(进行中):初始状态,操作正在进行中,尚未完成。
  2. Fulfilled(已完成):操作已经成功完成。
  3. Rejected(已失败):操作失败或出错。

Promise的创建通常使用new Promise()构造函数,并传入一个执行器函数作为参数。在这个执行器函数中,你通常会执行一些异步操作,并根据操作的结果调用resolve或reject函数。

Promise提供了.then()、.catch()和.finally()等方法来处理异步操作的结果或错误。.then()方法用于指定当Promise状态变为Fulfilled时要执行的回调函数,.catch()方法用于指定当Promise状态变为Rejected时要执行的回调函数,而.finally()方法则无论Promise状态如何都会执行。

Promise还有一些静态方法,如Promise.all()、Promise.race()、Promise.resolve()和Promise.reject()等,它们提供了对多个Promise对象进行操作的功能。

使用Promise可以简化异步代码的结构,避免回调地狱(Callback Hell)问题,并使代码更具可读性和可维护性。同时,Promise也支持链式调用和错误冒泡,使得异步操作的处理更加灵活和方便。

Promise在多种应用场景中都发挥了重要作用,特别是与异步操作相关的场景。以下是Promise主要解决的一些应用场景:

  1. 异步请求:在前端开发中,经常需要进行异步请求,如发送HTTP请求获取数据。使用Promise可以更好地处理这些异步请求,通过Promise的链式调用可以更清晰地表达异步操作之间的依赖关系。
  2. 定时器:在前端开发中,常常需要进行定时操作,如使用setTimeout。Promise也可以用于处理与定时器相关的异步操作。
  3. 数据库操作:当进行异步数据库查询时,可以使用Promise来封装查询操作,并在查询完成后返回结果或处理错误。
  4. 并行处理:使用Promise.all可以并行处理多个异步操作,并等待它们全部完成。这在需要同时执行多个异步任务,并等待所有任务完成后进行后续操作的场景中非常有用。
  5. 错误处理:Promise提供了.catch()方法来捕获异步操作中的错误,这使得错误处理更加方便。与传统的回调函数相比,Promise的错误处理更加直观和易于管理。
  6. 复杂的异步操作流程控制:在涉及多个异步操作,且这些操作之间存在复杂依赖关系的场景中,使用Promise可以更容易地控制异步操作的执行流程。通过Promise的链式调用和组合使用,可以构建出清晰、可维护的异步代码结构。

总之,Promise在处理异步操作、简化异步代码结构、避免回调地狱等方面都发挥了重要作用。它使得异步编程更加灵活、方便和易于管理。

宏任务有哪些

宏任务是指消息队列中的等待被主线程执行的事件。这些事件在JavaScript中主要涉及到一些异步操作,当这些异步操作完成时,它们的结果(通常是一个回调函数)会被添加到宏任务队列中等待处理。

以下是一些常见的宏任务类型:

  1. 整体代码script:这是最常见的宏任务,即主线程执行的代码本身。
  2. setTimeout:这是JavaScript中的一个函数,用于在指定的毫秒数后执行代码。当setTimeout的回调函数被触发时,它会被添加到宏任务队列中等待执行。
  3. setInterval:与setTimeout类似,但setInterval会定期(根据指定的时间间隔)执行回调函数。同样,当setInterval的回调函数被触发时,它也会被添加到宏任务队列中等待执行。
  4. setImmediate(Node.js环境):在Node.js环境中,setImmediate函数用于在I/O操作完成后立即执行一个函数。这个函数的回调函数也会被添加到宏任务队列中。
  5. requestAnimationFrame:这是一个浏览器提供的API,用于在下次浏览器重绘之前执行动画相关的代码。当requestAnimationFrame的回调函数被触发时,它也会被添加到宏任务队列中。
  6. UI渲染:在某些情况下(如浏览器环境),UI渲染也可以被视为一个宏任务。当页面需要更新或重绘时,浏览器会触发一个UI渲染的宏任务。

请注意,这些宏任务的执行顺序取决于JavaScript的事件循环机制。当所有同步代码执行完毕后,主线程会不断地从宏任务队列中取出任务并执行,直到所有任务都执行完毕。在这个过程中,如果遇到异步微任务(如Promise的then方法),它们会被添加到微任务队列中,并在当前宏任务执行完毕后立即执行。

微任务类型有哪些

在JavaScript中,微任务(MicroTask)是一种小的异步任务,它们在宏任务(MacroTask)执行完毕后立即执行。微任务通常包括以下几种类型:

  1. Promise:Promise是JavaScript中用于处理异步操作的对象。当Promise的状态改变(从pending变为fulfilled或rejected)时,其关联的then或catch方法中的回调函数会被添加到微任务队列中等待执行。Promise本身是同步的,但其then和catch回调是异步的微任务。
  2. process.nextTick(Node.js环境):在Node.js中,process.nextTick方法用于在当前执行栈的末尾添加一个回调函数,该函数会在当前宏任务(如事件循环的某个阶段)的末尾、但在任何微任务之前执行。这意味着process.nextTick的回调会先于Promise的then和catch回调执行。
  3. MutationObserver(浏览器环境):MutationObserver是一个浏览器提供的API,用于监视DOM树的变化(如添加、删除或修改节点)。当DOM树发生变化时,MutationObserver的回调函数会被添加到微任务队列中等待执行。
  4. Object.observe(已废弃):虽然Object.observe已经被废弃,但它也曾是一个用于监视JavaScript对象变化的API。当对象发生变化时,其关联的回调函数会被添加到微任务队列中等待执行。然而,由于性能和复杂性等问题,这个API已经被废弃,取而代之的是Proxy对象。
  5. async/await:async/await是JavaScript中处理异步操作的另一种方式。async函数返回一个Promise对象,而await表达式会暂停当前async函数的执行并等待Promise解析完成。当Promise解析完成后,其结果会被返回给await表达式,并继续执行后续的异步代码。在这个过程中,await后的代码实际上是作为一个微任务来执行的。

请注意,微任务的执行顺序是先进先出(FIFO)的,即先添加到微任务队列中的任务会先被执行。此外,在每次事件循环中,微任务队列都会被清空,即所有微任务都会被执行完毕后再开始下一个宏任务。

js的事件代理

数据更新时,shitou

在JavaScript中,事件代理(也称为事件委托或事件委托传播)是一种技术,它允许你将事件监听器添加到一个父元素或祖先元素上,而不是直接添加到目标元素上。然后,当事件在子元素(目标元素)上触发时,事件会冒泡到父元素或祖先元素,从而触发绑定在它们上面的事件监听器。

使用事件代理的好处包括:

  1. 节省内存:你不需要为每个元素都添加一个事件监听器,只需在父元素上添加一个即可。
  2. 动态内容:如果子元素是动态添加的(例如,通过AJAX请求),那么直接绑定在它们上面的事件监听器将不会工作。但是,由于事件监听器是绑定在父元素上的,所以新添加的子元素仍然可以触发事件。
  3. 简化代码:你可以将多个事件监听器合并到一个父元素上,而不是分别绑定到每个子元素上。

下面是一个使用事件代理的简单示例:

javascript复制代码

// 假设你有一个包含多个按钮的div

// <div id="buttonContainer">

// <button class="myButton">Button 1</button>

// <button class="myButton">Button 2</button>

// <!-- ... 其他按钮 ... -->

// </div>

// 使用事件代理在buttonContainer上添加一个点击事件监听器

document.getElementById('buttonContainer').addEventListener('click', function(event) {

// 检查被点击的元素是否是我们关心的元素(即带有myButton类的按钮)

if (event.target.classList.contains('myButton')) {

// 如果是被关心的元素,执行相应的操作

alert('You clicked a button!');

}

});

在这个示例中,我们只在buttonContainer元素上添加了一个点击事件监听器。但是,当用户点击任何带有myButton类的按钮时,都会触发这个事件监听器。这是通过检查event.target(即被点击的元素)来实现的。如果event.target是我们关心的元素(即带有myButton类的按钮),那么就执行相应的操作。

js继承,方式,优缺点

在JavaScript中,有多种方式可以实现继承,每种方式都有其优缺点。以下是几种常见的继承方式及其优缺点:

1. 原型链继承

优点:

实现简单,易于理解。

父类新增的属性和方法,子类都能访问到。

缺点:

创建子类实例时,不能向父类构造函数传参。

所有子类实例都会共享父类实例的属性(如果是引用类型),这可能导致修改一个子类实例的属性而影响到另一个子类实例。

在子类型中,无法重写父类型的引用类型属性,这是由于共享了地址的引用类型属性。

2. 借用构造函数(类式继承、伪经典继承)

优点:

解决了原型链继承中子类实例共享父类引用类型属性的问题。

可以在子类型构造函数中向超类型构造函数传参。

缺点:

方法都在构造函数中定义,因此函数复用就无从谈起。

在超类型原型中定义的方法,对子类型而言也是不可见的,结果所有类型都只能使用构造函数模式。

在超类型构造函数中定义的属性,对子类型而言也是不可见的,结果就是所有实例都会继承到相同的属性。

3. 组合继承(原型链+借用构造函数)

优点:

融合了原型链继承和构造函数继承的优点,是JavaScript中最常用的继承模式。

它既能使用实例属性,也能继承原型属性。

在子类型构造函数中,可以调用超类型构造函数,因此可以继承超类型的属性。

同时还可以保证每个实例都有它自己的属性。

缺点:

调用了两次父类构造函数,生成了两份实例(子类实例将子类原型上的那份属性屏蔽了)。

4. 原型式继承

优点:

无需创建构造函数,可以基于已有的对象创建新对象。

将已有对象作为新创建对象的原型,新对象会继承原型对象的所有属性和方法。

缺点:

包含引用类型的属性值始终都会共享相应的值,这点跟原型链继承一样。

5. 寄生组合式继承

优点:

通过借用构造函数来继承属性,通过原型链的混成形式来继承方法。

不调用两次父类构造函数,避免在子类原型上创建不必要的、多余的属性。

基本上,子类可以继承父类的所有属性和方法,同时避免了上述三种模式的缺点。

缺点:

实现起来相对复杂一些。

6. ES6中的类继承

优点:

语法更接近于传统面向对象编程语言的类继承。

提供了class、extends、super等关键字,使得继承的实现更加简洁和直观。

支持静态方法和静态属性。

缺点:

底层仍然是基于原型链和构造函数继承实现的,只是语法上做了封装。

对于习惯了传统面向对象编程语言的开发者来说,可能需要一些时间来适应这种新的继承方式。

js闭包

在JavaScript中,闭包(Closure)是一个强大的特性,它允许函数记住并访问其词法作用域(lexical scope),即使函数在其词法作用域之外执行。简单来说,闭包就是一个函数可以访问其自己的作用域(词法环境)以及其父级(词法上封闭)的作用域中的变量的函数。

闭包的特点

函数内部可以访问定义它的函数外部的变量:即使外部函数已经执行完毕,其内部变量依然可以被内部函数访问。

变量不会被垃圾回收:由于闭包可以访问其外部函数的变量,因此只要闭包存在,其外部函数的变量就不会被垃圾回收。

封装私有变量:闭包可以创建私有变量,这些变量只能通过特定的公开方法进行访问和修改。

闭包的示例

javascript复制代码

function outerFunction(outerVariable) {

return function innerFunction(innerVariable) {

console.log('outerVariable:', outerVariable);

console.log('innerVariable:', innerVariable);

};

}

const myClosure = outerFunction('Hello from outer');

myClosure('Hello from inner'); // 输出: outerVariable: Hello from outer, innerVariable: Hello from inner

在上面的示例中,innerFunction是一个闭包,因为它可以访问其外部函数outerFunction的变量outerVariable。即使outerFunction已经执行完毕,outerVariable依然可以被innerFunction访问。

闭包的用途

  1. 数据封装和私有变量:通过闭包,我们可以实现数据的封装,将某些变量限制在特定的作用域中,只能通过特定的方法进行访问和修改。
  2. 回调函数和事件处理程序:在JavaScript中,我们经常使用回调函数来处理异步操作(如Ajax请求)或事件(如按钮点击)。闭包可以确保回调函数可以访问其定义时所在的作用域中的变量。
  3. 实现模块和组件:通过闭包,我们可以创建具有私有变量和公开方法的模块或组件,实现代码的模块化和组件化。

注意事项

虽然闭包在JavaScript中非常有用,但过度使用闭包也可能导致一些问题,如内存泄漏和性能问题。因此,在使用闭包时,我们需要注意以下几点:

  1. 避免不必要的闭包:如果不需要访问外部函数的变量,就不要创建闭包。
  2. 注意内存管理:由于闭包可以阻止其外部函数的变量被垃圾回收,因此需要注意不要创建过多的闭包,以免导致内存泄漏。
  3. 谨慎使用全局变量:在闭包中访问和修改全局变量可能会导致意外的副作用和难以调试的问题。因此,我们应该尽量避免在闭包中访问和修改全局变量。

内存泄露

内存泄露(Memory Leak)是指在程序运行过程中,错误地保持了本应该释放的对象或内存块,导致系统无法重新分配这部分内存给其他对象使用,进而造成内存的浪费,甚至可能导致程序运行缓慢、崩溃或系统资源耗尽。

内存泄露的常见原因包括:

长生命周期的对象持有短生命周期对象的引用:当一个对象的生命周期比它持有的引用对象的生命周期长时,如果长生命周期对象一直持有短生命周期对象的引用,那么短生命周期对象将无法被垃圾回收机制回收,造成内存泄露。

缓存不当:缓存对象过多且没有适当的清理机制,会导致内存占用持续增长,最终造成内存泄露。

事件监听器未移除:在JavaScript等事件驱动型语言中,如果注册了事件监听器但未在适当的时候移除,那么当不再需要这些监听器时,它们仍然会占用内存。

闭包使用不当:在JavaScript中,闭包可以保持对外部作用域的引用,如果闭包被长期持有,那么外部作用域中的对象也将无法被回收。

全局变量:在全局作用域中声明的变量不会被垃圾回收机制自动回收,如果频繁地创建全局变量,会导致内存泄露。

资源未释放:如文件句柄、数据库连接等系统资源,在使用完毕后需要显式地关闭或释放,否则可能导致内存泄露。

如何检测和修复内存泄露:

使用内存分析工具:内存分析工具可以监控程序的内存使用情况,帮助开发者发现内存泄露的源头。

代码审查:对代码进行仔细的审查,特别是涉及到对象生命周期管理和资源释放的代码部分。

单元测试:编写针对内存泄露的单元测试,通过模拟长时间运行的场景来检测内存泄露。

修复泄露:一旦确定了内存泄露的源头,就需要修改代码来修复它,确保不再持有不再需要的对象的引用,并及时释放资源。

需要注意的是,内存泄露是一个复杂的问题,需要仔细分析和调试才能找到并修复。在开发过程中,应该尽量避免使用可能导致内存泄露的编程模式,并时刻关注程序的内存使用情况。

Js箭头函数的特点

  1. 没有自己的 this:箭头函数不会创建自己的 this 上下文,所以 this 值会捕获其所在上下文的 this 值。这是箭头函数在处理回调函数和事件处理器时特别有用的一个特点,因为在这些情况下,this 的值经常会被意外地改变。
  2. 没有 arguments 对象:箭头函数没有 arguments 绑定。如果你需要访问函数参数列表内的参数,你可以使用剩余参数(rest parameters)...。
  3. 不能用作构造函数:因为箭头函数没有自己的 this,所以它们不能被用作构造函数,使用 new 调用它们会抛出错误。
  4. 没有 prototype 属性:由于箭头函数不能用作构造函数,所以它们也没有 prototype 属性。
  5. 更少的语法糖:箭头函数在语法上比传统函数更简洁,尤其是在处理简单函数或回调函数时。

定时器函数

1、setInterval()是JavaScript的一个定时器函数,用于设置周期性的定时任务。

setInterval(function, delay, param1, param2, …);

  1. 其中,function参数表示要执行的函数,delay参数表示定时器的时间间隔(以毫秒为单位),param1, param2, …表示传递给函数的参数(可选)。
  2. setInterval()会不断地重复执行指定的函数,直到被取消或页面被卸载。每个时间间隔结束时,function函数都会被调用一次。

2、setTimeout也是一个定时器函数,它允许我们在指定的时间后执行一次函数。

setTimeout(function, delay, arg1, arg2, …)

  1. 其中,function 是我们要执行的函数,delay 是延迟的时间(以毫秒为单位),arg1、arg2 等是传递给函数的参数(可选)。
  2. setTimeout 返回一个唯一的标识符,我们可以使用 clearTimeout 函数来取消这个定时器。
  1. 区别
  1. setInterval() 在延迟指定时间后重复执行指定任务,直到被取消或页面关闭。而 setTimeout() 在延迟指定时间后执行指定任务,只执行一次;
  2. setInterval() 的执行间隔时间是固定的,而 setTimeout() 可以动态调整延迟时间;
  3. setInterval() 可能会存在累积性误差,因为它的执行时间是相对于上一次任务结束的时间计算的,如果执行的任务耗时超过了指定的时间间隔,就会出现累积性误差。而 setTimeout() 每次执行都是相对于上一次任务开始的时间计算的,不存在累积性误差;
  4. setInterval() 可能会对性能产生影响,因为它会不断地重复执行指定任务,占用 CPU 资源。而 setTimeout() 只会在指定时间后执行一次任务,对性能影响较小。
  1. 注意事项

使用setInterval()时要注意以下几点:

  1. delay参数的最小值是4毫秒,如果设置的值小于4毫秒,则会被强制转换为4毫秒。
  2. setInterval()返回一个定时器ID,可以使用clearInterval()函数来取消定时器。
  3. 由于JavaScript是单线程的,因此如果function函数执行的时间过长,可能会影响页面的响应性能。因此,建议在编写function函数时要尽量减少执行时间。

Js 高阶函数 reduce()

reduce() 是 JavaScript 中的一个高阶函数,它属于数组的原型方法。这个函数对数组中的每个元素执行一个提供的 reducer 函数(升阶函数),将其结果汇总为单个输出值。reduce() 的工作方式是从左到右遍历数组,将数组的每个值(或初始值)减少到一个单个值。

基本语法如下:

javascript复制代码

arr.reduce(callback(accumulator, currentValue[, index[, array]])[, initialValue])

callback:一个执行器中调用的函数,接收四个参数:

accumulator(累加器):累计器累加回调的返回值; 它是上一次调用回调时返回的累积值,或提供的初始值(initialValue)。

currentValue(当前值):数组中正在处理的当前元素。

index(可选):数组中正在处理的当前元素的索引。如果提供了 initialValue,则索引从0开始;否则从1开始。

array(可选):调用 reduce 的数组。

initialValue(可选):作为第一次调用 callback 函数时的第一个参数的值。如果没有提供初始值,则将使用数组中的第一个元素。在没有初始值的空数组上调用 reduce 将报错。

示例:

计算数组中所有数字的总和:

javascript复制代码

const numbers = [1, 2, 3, 4];

const sum = numbers.reduce((accumulator, currentValue) => accumulator + currentValue, 0);

console.log(sum); // 输出 10

计算数组中所有数字的平均值:

javascript复制代码

const numbers = [1, 2, 3, 4];

const sum = numbers.reduce((accumulator, currentValue) => accumulator + currentValue, 0);

const avg = sum / numbers.length;

console.log(avg); // 输出 2.5

将二维数组展平为一维数组:

javascript复制代码

const array2D = [[1, 2], [3, 4], [5, 6]];

const array1D = array2D.reduce((accumulator, currentValue) => accumulator.concat(currentValue), []);

console.log(array1D); // 输出 [1, 2, 3, 4, 5, 6]

(注意:在 ES6 中,你还可以使用 flat() 方法来实现二维数组的展平,如 array2D.flat()。)

reduce() 是一个非常有用的函数,它允许你根据自定义的逻辑对数组元素进行累积操作。

递归调用方法的关键

递归调用方法的关键在于以下几个要点:

  1. 明确递归的终止条件:
  1. 这是递归能够正确停止的关键。必须有一个或多个条件,当满足这些条件时,递归函数将不再调用自身,而是直接返回结果。
  2. 例如,在计算阶乘的递归函数中,当 n 等于 0 或 1 时,函数直接返回 1,而不是继续递归。
  1. 递归表达式:
  1. 递归表达式定义了如何将问题分解为更小的子问题,并如何使用子问题的解来求解原问题。
  2. 它通常涉及对当前参数进行一些处理,并递归地调用函数自身,但使用不同的参数(通常是当前参数的一个简化版本)。
  1. 参数传递:
  1. 在递归调用中,需要确保正确地将参数传递给下一次递归调用。
  2. 参数可能会逐渐接近递归的终止条件,或者可能会通过某种方式累积结果。
  1. 避免栈溢出:
  1. 由于递归是通过函数调用的栈来实现的,如果递归过深,可能会导致栈溢出。
  2. 因此,需要确保递归的深度是可控的,或者使用其他技术(如尾递归优化)来减少栈的使用。
  1. 可读性和可维护性:
  1. 递归代码有时可能难以理解和维护。因此,编写递归函数时,应注意代码的清晰性和可读性。
  2. 可以使用注释来解释递归的逻辑和每个步骤的作用。
  1. 考虑迭代替代方案:
  1. 虽然递归在某些情况下非常有用,但并非所有问题都适合使用递归解决。
  2. 在某些情况下,迭代方法可能更加高效或更容易理解。因此,在决定使用递归之前,应仔细考虑问题的性质和要求。
  • 11
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

水&陌&殇

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值