javascript如何运行_浏览器背后发生了什么

8次阅读

JavaScript 由浏览器引擎(如 V8)解析执行:先词法分析、语法分析生成 AST,再 JIT 编译为机器码运行;script 放 body 底部可避免阻塞 DOM 解析;defer 确保 DOM 就绪后执行,async 则执行时机不可控;函数调用创建执行上下文并入栈;eval 和 Function 因动态编译、安全风险及导致 JIT 失效而应避免。

javascript 如何运行_浏览器背后发生了什么

JavaScript 代码怎么被 浏览器 执行的

浏览器不是直接“运行”JavaScript,而是通过引擎把文本代码一步步解析、编译、执行。主流浏览器用的引擎不同:V8(Chrome、Edge、Node.js)、SpiderMonkey(Firefox)、JavaScriptCore(Safari),但流程高度一致。

简单说:你写的 console.log("hello") 会经历——读取 HTML 中的 标签 → 提取 JS 字符串 → 词法分析(拆成 console.log 等 token)→ 语法分析(生成抽象语法树 AST)→ 解释执行或即时编译(JIT)为机器码 → 调用 V8 的 Runtime 接口输出到控制台。

为什么 script 放在 body 底部更安全

因为浏览器解析 HTML 是自上而下流式进行的,遇到 会暂停 DOM 构建,去下载、解析、执行 JS —— 这期间页面渲染卡住,用户看到白屏。如果脚本里写了 document.getElementById("app"),而 #app 在它下面还没解析到,结果就是 null

  • 使用 defer:脚本并行下载,等 DOM 解析完、DOMContentLoaded 前执行,保证 DOM 可访问
  • 使用 async:下载不阻塞,但执行时机不可控,可能 DOM 还没就绪
  • 手动监听 DOMContentLoaded 事件,确保 DOM 已加载完成再操作节点
document.addEventListener('DOMContentLoaded', () => {const el = document.getElementById('main');   if (el) el.innerHTML = 'Ready'; });

V8 引擎里函数调用实际发生了什么

每次调用函数,V8 都会创建一个执行上下文(Execution Context),压入调用栈(Call Stack)。这个上下文包含:VariableEnvironment(var 声明)、LexicalEnvironment(let/const)、this 绑定、outer environment reference(用于闭包)。

立即学习Java 免费学习笔记(深入)”;

常见误解是“函数定义时就确定了作用域”,其实没错,但关键点在于:闭包捕获的是变量的引用,不是值。比如循环中用 var 定义 i,所有回调共享同一个 i;换成 let 就每轮迭代生成独立绑定。

for (var i = 0; i < 3; i++) {setTimeout(() => console.log(i), 0); // 输出 3, 3, 3 } for (let j = 0; j < 3; j++) {setTimeout(() => console.log(j), 0); // 输出 0, 1, 2 }

eval 和 Function 构造函数为什么危险又慢

eval()new Function(……) 会触发完整的编译流程,且无法被 V8 的优化编译器(TurboFan)内联或优化。更重要的是,它们能访问当前作用域 —— 如果传入的字符串来自用户输入(比如表单、URL 参数),就等于开了远程代码执行后门。

  • eval("x + 1") 必须重新解析、生成 AST、生成 字节 码,跳过所有缓存
  • new Function("return" + userInput)() 不共享外层词法环境,相对安全一点,但仍绕过静态分析和 CSP(Content Security Policy)
  • 现代框架(如 React)用 Babel 编译 JSX,绝不用 eval 动态执行模板字符串

真正难察觉的是 JIT 优化失效:一个高频函数如果内部用了 eval,V8 会直接标记为“deoptimized”,降级回解释执行,性能可能跌一个数量级。

text=ZqhQzanResources