JavaScript 底层:引擎、作用域与变量那点事儿
你可以把 JavaScript 运行的全过程,想象成引擎在 搭积木,按规矩进行 “观察→拆积木→搭积木→正式运行”。
一、JS 引擎:浏览器和 Node 背后的“超级大脑”
JavaScript 本身不会自己跑,它得靠 引擎 才能动起来。
主流 JS 引擎:
- 浏览器 :V8 引擎
- Node.js :V8 引擎
可以理解成: 浏览器是书房,Node.js 是工作室,但它们是同一个超级大脑在工作——V8。
二、JS 代码是怎么跑起来的?
很多人以为代码一写就跑,其实 V8 宝宝超有耐心,它会按四步慢慢来:
- 先读一遍 : V8 拿到代码,不着急跑,先整体看一眼。
- 分词(拆积木) : 把一整行代码拆成一小块一小块:
var a = 10→ 拆成var、a、=、10。 - 语法分析(搭积木) : 把积木拼成一棵树,叫 AST 抽象语法树。 简单说:给代码排排队、分分类、理逻辑。
- 正式执行 : 树搭好了,开始一条一条运行。
一句话总结: V8 不莽撞,先梳理、再分析、最后才执行。
三、作用域:变量的“活动地盘”
作用域就是变量能在哪儿活动、在哪儿说话算数的地盘。
一共有 3 种:
- 全局作用域 : 最大最外层,谁都能访问,像“小区广场”。
var a = 10;
function foo(){
console.log(a);
}
foo(); // 10
console.log(a); // 10
- 函数作用域 : 函数里面的小空间,外面进不来,像“家里客厅”。
function foo(){ //函数作用域
var a = 10 ;
console.log(a); // 10
}
function foo(){ //函数作用域
var a = 10 ;
}
console.log(a); // 报错 a is not defined
- 块级作用域 : 大括号
{}包起来的地方,像“卧室”。
{
let a = 10;
console.log(a); // 10
}
{
let a = 10;
}
console.log(a); // 报错 a is not defined
特点:写代码时就定好了,运行时改不了。
四、v8怎么从作用域找变量?像找妈妈
V8 宝宝找变量特别乖,一层一层往上问:
- 先在自己房间找
- 找不到去客厅找
- 再找不到去小区广场找
- 还找不到 → 直接报错:变量不存在!
// 全局作用区: 小区广场找不到
function foo(){ // 函数作用区: 客厅内找不到
if(true){
console.log(a); // 块级作用区: 房间内找不到
}
}
// 最终 报错:a is not defined
这就叫 作用域链。
五、let 和 const:现代 JS 的“乖宝宝声明”
var
// var声明在全局作用域、提升、重复声明
console.log(a);
{
var a = 10;
var a =20;
}
// undefined
这个 var 太调皮,到处乱窜。 ES6 来了两个乖宝宝:let、const。
let
- 碰到
{}就形成 块级作用域,关在里面不乱跑。
{
let a = 10;
}
console.log(a); // 报错:a is not defined
- 不提升,必须先声明再使用。
console.log(a);
let a = 10;
// 报错:Cannot access 'a' before initialization
- 不能重复声明,避免搞混。
{
let a = 10;
let a =20;
// 编辑器直接报错
}
const
- 常量,一旦赋值不能改。
const PI = 3.1415926
PI = 3
console.log(PI);// Assignment to constant variable.
- 必须声明时就赋值。
const PI
PI = 3.1415926
console.log(PI);// 编辑器直接报错
总而言之
JS 并不是简单的脚本语言,它背后有严谨的编译与执行规则;V8 引擎像一位细心的管理员,先梳理再运行;作用域像变量的“活动区域”,让代码更安全有序;let 和 const 则让变量声明更规范、更可靠。