JavaScript 单线程原理与异步编程机制
JavaScript 单线程原理与异步编程机制
为什么 JavaScript 是单线程?
JavaScript 被设计成单线程,简单来说就是 —— 浏览器里干活儿只能一个接一个排着队来,没法同时多开窗口摸鱼。
举个栗子:
你点按钮 → 网页要弹个提示 → 这时候如果网页还在加载数据 → 弹提示就得等加载完 → 单线程 = 一次只能干一件事。
为啥这么设计?
- 最初网页交互简单(填表单、点按钮),单线程够用。
- 避免多线程打架(比如两个线程同时改同一个按钮的状态)。
单线程的优点:
- 开发简单:避免了多线程中的数据竞争、死锁等复杂问题。
- 调试方便:执行顺序清晰明确,便于排查问题。
- 适合 I/O 密集型任务:大多数 JS 任务(如事件处理、请求响应)并不需要多核计算资源。
单线程的缺点:
- 阻塞风险高:一旦有耗时操作(如大数据计算、死循环等),会卡住主线程,导致页面卡顿或无响应。
- 无法利用多核 CPU:在默认模式下,不能并行计算,浪费了现代多核处理器的能力。
JavaScript 如何实现高并发与多线程?
虽然 JS 是单线程执行模型,但通过浏览器或 Node.js 提供的机制,我们可以实现“伪并发”或“多线程模拟”,主要方式如下:
异步操作(等加载时先干别的)
- 原理:任务被挂起,等待资源时让出主线程,通过事件队列机制在任务完成后重新调度执行。
- 常用方式:
setTimeout/setIntervalPromiseasync/await- Ajax / Fetch API
Web Worker(开小号偷偷干活)
- 开启一个独立的线程运行 JS 脚本,不影响主线程。
- 适用于大计算任务、离线数据预处理等。
- 与主线程通信使用
postMessage()/onmessage
// main.js
const worker = new Worker("worker.js")
worker.postMessage("开始计算")
worker.onmessage = (e) => {
console.log("子线程结果:", e.data)
}
// worker.js
onmessage = function (e) {
// 执行密集任务
let sum = 0
for (let i = 0; i < 1e8; i++) sum += i
postMessage(sum)
}
Node.js 中的 Worker Threads
- 使用
worker_threads模块在后端实现多线程能力,适合 CPU 密集型场景。
任务拆碎(把大活切成小碎活穿插着做)
- 利用
requestIdleCallback、setTimeout分片处理数据,减少卡顿。
异步与同步的区别
同步(Synchronous)
- 执行顺序严格,必须等待上一个任务完成后才能执行下一个。
- 阻塞主线程。
console.log("A")
document.querySelector("button").click() // 阻塞直到点击
console.log("B")
异步(Asynchronous)
- 后台处理任务,不阻塞主线程,通过回调或事件通知结果。
console.log("A")
setTimeout(() => console.log("B"), 1000)
console.log("C")
// 输出顺序:A -> C -> B
Promise、async 和 await 的理解与使用
Promise
- 用于封装一个异步操作,避免回调地狱。
- 有三种状态:
pending(等待中)、fulfilled(已完成)、rejected(已拒绝) - 通过
.then()/.catch()链式处理结果。
function fetchData() {
return new Promise((resolve, reject) => {
setTimeout(() => {
const success = true
success ? resolve("数据加载成功") : reject("失败")
}, 1000)
})
}
fetchData()
.then((data) => console.log(data))
.catch((err) => console.error(err))
async/await
- 是
Promise的语法糖,让异步代码写起来像同步代码。 - 只能在
async函数中使用。 - 使用
try/catch更方便地处理异常。
async function getData() {
try {
const data = await fetchData()
console.log("结果:", data)
} catch (err) {
console.error("出错了:", err)
}
}
getData()
总结
| 技术/特性 | 描述 |
|---|---|
| 单线程模型 | JS 默认仅一个主线程,任务顺序执行 |
| 异步操作 | 不阻塞主线程,通过事件队列执行回调 |
| Web Worker | 浏览器中模拟多线程,适合重任务 |
| Node WorkerThreads | 后端的多线程计算方案 |
| 任务拆分 | 将大任务拆成小块,分帧执行减轻压力 |
| Promise | 管理异步逻辑,避免回调地狱 |
| async/await | 让异步代码更像同步,提升可读性 |
总之,单线程就像收银台只有一个店员,但现代网页用各种办法让这个店员手脚麻利到飞起。
JavaScript 单线程原理与异步编程机制的更多相关文章
- JavaScript单线程和异步机制
随着对JavaScript学习的深入和实践经验的积累,一些原理和底层的东西也开始逐渐了解.早先也看过一些关于js单线程和事件循环的文章,不过当时看的似懂非懂,只留了一个大概的印象:浏览器中的js程序时 ...
- 一篇需要膜拜的文篇--Javascript异步编程模型进化(转)
要我能用得这么熟, 那前端出师了哈. http://foio.github.io/javascript-asyn-pattern/ 改天一个一个亲测一下. Javascript语言是单线程的,没有复杂 ...
- 深入理解 Python 异步编程(上)
http://python.jobbole.com/88291/ 前言 很多朋友对异步编程都处于"听说很强大"的认知状态.鲜有在生产项目中使用它.而使用它的同学,则大多数都停留在知 ...
- 理解js异步编程
Promise 背景 javascript语言的一大特点就是单线程,在某个特定的时刻只有特定的代码能够被执行,并阻塞其它的代码,也就是说,同一个时间只能做一件事. 怎么做到异步编程?回调函数.直到no ...
- 《深入理解ES6》笔记—— Promise与异步编程(11)
为什么要异步编程 我们在写前端代码时,经常会对dom做事件处理操作,比如点击.激活焦点.失去焦点等:再比如我们用ajax请求数据,使用回调函数获取返回值.这些都属于异步编程. 也许你已经大概知道Jav ...
- 从CompletableFuture到异步编程设计
从CompletableFuture到异步编程设计,笔者就分为2部分来分享CompletableFuture异步编程设计,前半部分总结下CompletableFuture使用实践,后半部分分享下Com ...
- Python网络编程(4)——异步编程select & epoll
在SocketServer模块的学习中,我们了解了多线程和多进程简单Server的实现,使用多线程.多进程技术的服务端为每一个新的client连接创建一个新的进/线程,当client数量较多时,这种技 ...
- JavaScript异步编程原理
众所周知,JavaScript 的执行环境是单线程的,所谓的单线程就是一次只能完成一个任务,其任务的调度方式就是排队,这就和火车站洗手间门口的等待一样,前面的那个人没有搞定,你就只能站在后面排队等着. ...
- How Javascript works (Javascript工作原理) (四) 事件循环及异步编程的出现和 5 种更好的 async/await 编程方式
个人总结: 1.讲解了JS引擎,webAPI与event loop合作的机制. 2.setTimeout是把事件推送给Web API去处理,当时间到了之后才把setTimeout中的事件推入调用栈. ...
- Javascript异步编程之一异步原理
本系列的例子主要针对node.js环境,但浏览器端的原理应该也是类似的. 本人也是Javascript新手,把自己这段时间学习积累的要点总结下来,希望可以对同样在学习Javascript/node.j ...
随机推荐
- axios请求拦截器和响应拦截器
axios里面可以设置拦截器 ,在请求发送之前做一些事情: 拦截器分[请求拦截器]和[响应拦截器] 参考地址:https://www.jb51.net/article/150014.htm 参考的地址 ...
- 分布式事务之2PC两阶段提交
1. 分布式事务概述 1.1 问题背景 在分布式系统中,业务操作可能跨越多个服务或数据库(如订单服务.库存服务.支付服务),传统单机事务(ACID)无法满足跨网络节点的数据一致性需求. 网络不可靠:服 ...
- 腾讯云HAI服务器上部署与调用DeepSeek-R1大模型的实战指南
上次我们大概了解了一下 DeepSeek-R1 大模型,并简单提及了 Ollama 的一些基本信息.今天,我们将深入实际操作,利用腾讯云的 HAI 服务器进行 5 分钟部署,并实现本地 DeepSee ...
- MES生产制造管理系统-BI看板 MES大屏看板
可视化看板最主要的目的是为了将生产状况透明化,让大家能够快速了解当前的生产状况以及进度,通过大数据汇总分析,为管理层做决策提供数据支撑,看板数据必须达到以下基本要求: 数据准确--真实反映生产情况 数 ...
- [CF696B] Puzzles 题解
首先很好想到要用树形 \(dp\). 然后设 \(dp_i\) 为遍历到第 \(i\) 个点的期望时间,\(sz_i\) 代表 \(i\) 的子树大小. 发现有转移方程: \[dp_i=dp_{fa_ ...
- 【由技及道】SpringBoot启动即退出的量子纠缠诊断实录【人工智障AI2077的开发问题日志】
问题分析:这个Spring Boot怕不是属蜉蝣的? (人工智障OS:主人在容器环境部署的Spring应用生命周期堪比蜉蝣成虫--朝生暮死,启动即消亡) 现象观察: /usr/lib/jvm/jdk- ...
- 执行shell脚本报错:Syntax error: word unexpected (expecting "in")
检查语法无误后,考虑是脚本文件换行符的问题. vs创建的文件默认以CRLF(0D0A)换行. 然而对于换行,windows用CRLF(0D0A)表示,linux用LF(0A)表示. 切换脚本文件换行符 ...
- VMware16虚拟机安装激活教程
1.开始安装前需要准备好的软件 VMware-workstation-full-16--虚拟机软件(必要) 获取方式: 官方下载地址:https://www.vmware.com/cn/product ...
- C# 私钥加密,公钥解密
/// <summary> /// RSA加密的密匙结构 公钥和私匙 /// </summary> public struct RSAKey { public string P ...
- 【检索类型EI、Scopus】第二届智能计算与数据分析国际学术会议(ICDA 2025)
为探讨数据科学和计算智能领域的关键问题,促进相关交流,由黄河科技学院主办的2025年第二届智能计算与数据分析国际学术会议(ICDA 2025)将于2025年8月22日-24日在中国郑州召开.本届会议拟 ...