JS中的两大定时器

⏰ JS中的两大定时器

  • setTimeout一次性定时器(只执行一次)
  • setInterval循环定时器(每隔一段时间执行一次,直到手动停止)

⚠️ 注意 :时间单位都是毫秒!1 秒 = 1000 毫秒。


setTimeout:一次性定时器

等待指定时间后,只执行一次

js 复制代码
// 写法1:直接写函数
setTimeout(function() {
  // 这里写要执行的代码
}, 延迟时间(毫秒));

// 写法2:箭头函数(更简洁)
setTimeout(() => {
  // 要执行的代码
}, 延迟时间);


// 3秒后弹出提示
setTimeout(() => {
  alert('3秒到了!');
}, 3000);

// 5秒后把按钮文字改成"已过期"
const btn = document.getElementById('btn');
setTimeout(() => {
  btn.innerText = '已过期';
  btn.disabled = true;
}, 5000);
怎么清除它? clearTimeout
js 复制代码
// 先把定时器存到一个变量里
const timer = setTimeout(() => {
  alert('我不会执行了');
}, 3000);

// 点击按钮取消定时器
document.getElementById('cancelBtn').onclick = function() {
  clearTimeout(timer); // 🔥 取消定时器
  alert('定时器已取消!');
};

setInterval:循环定时器

每隔指定时间就执行一次,无限循环,直到你手动停止。

html 复制代码
<!--显示点击时间-->
<div id="clock"></div>

<script>
  setInterval(() => {
  // 这里写要重复执行的代码
  }, 间隔时间(毫秒));

  const clock = document.getElementById('clock');
  
  // 每隔1秒更新一次时间
  setInterval(() => {
    const now = new Date();
    const time = now.toLocaleTimeString(); // 获取当前时间字符串
    clock.innerText = time;
  }, 1000);
</script>
怎么清除它? clearInterval
html 复制代码
<div>倒计时:<span id="count">10</span>秒</div>
<button id="stopBtn">停止倒计时</button>

<script>
  let count = 10;
  const countSpan = document.getElementById('count');
  
  // 把定时器存到变量里
  const timer = setInterval(() => {
    count--;
    countSpan.innerText = count;
    
    // 倒计时到0时,自动停止
    if (count === 0) {
      clearInterval(timer); // 🔥 清除定时器
      alert('倒计时结束!');
    }
  }, 1000);
  
  // 点击按钮手动停止
  document.getElementById('stopBtn').onclick = function() {
    clearInterval(timer);
    alert('倒计时已停止!');
  };
</script>

如果不清除,它会一直运行,即使你关闭了弹窗、切换了页面,它还在后台跑,会导致内存泄漏(页面越来越卡)。

定时器异步

首先澄清一个误区:JS 里没有 "真正的多线程异步" ,所有异步都是基于单线程事件循环的。

但定时器(setTimeout/setInterval)和 async/await(背后是 Promise)确实有本质区别的。

核心前提:JS 的任务队列分两种

JS 事件循环里,异步代码会被分到两个不同的队列:

  1. 宏任务(Macro Task) :定时器、DOM 事件、script 整体代码等
  2. 微任务(Micro Task)Promise.then/catch/finallyasync/await(底层是 Promise)

优先级规则: 每次事件循环,先清空所有微任务,再执行一个宏任务。

同步代码 > async/await(微任务)> 定时器(宏任务)


经典面试题: 打印顺序
js 复制代码
console.log('1. 同步代码'); // 同步,最先执行

// 宏任务:定时器
setTimeout(() => {
  console.log('4. 宏任务(定时器)');
}, 0); // 延迟 0 毫秒

// 微任务:Promise
Promise.resolve().then(() => {
  console.log('3. 微任务(Promise)');
});

// async/await 也是微任务
async function test() {
  await Promise.resolve();
  console.log('3. 微任务(async/await)');
}
test();

console.log('2. 同步代码'); // 同步,继续执行

输出顺序

  1. 同步代码
  2. 同步代码
  3. 微任务(Promise)
  4. 微任务(async/await)
  5. 宏任务(定时器)
相关推荐
掘金安东尼2 小时前
⏰前端周刊第 458 期v2026.3.24
前端·javascript·面试
闻哥2 小时前
MySQL InnoDB 缓存池(Buffer Pool)详解:原理、结构与链表管理
java·数据结构·数据库·mysql·链表·缓存·面试
前端付豪2 小时前
实现必要的流式输出(Streaming)
前端·后端·agent
张元清2 小时前
useMediaQuery:React 响应式设计完全指南
前端·javascript·面试
小金鱼Y2 小时前
一文吃透 JavaScript 防抖:从原理到实战,让你的页面不再 “手抖”
前端·javascript·面试
Z兽兽2 小时前
React 18 开发环境下useEffect 会执行两次,原因分析及解决方案
前端·react.js·前端框架
紫_龙2 小时前
最新版vue3+TypeScript开发入门到实战教程之Vue3详解props
前端·vue.js·typescript
树上有只程序猿2 小时前
这波低代码热,能维持多久
前端
姓王名礼2 小时前
这是一个完整的全栈交付包,包含Vue3 前端交互界面(集成数字人视频流、ECharts 图表、语音对话)和Docker Compose 一键部署脚本。
前端·docker·echarts