简单理解

var zm = function (x) {
var code = 'bb'
return code
};

学过js的老哥们都知道,当这样简单的一个函数进入浏览器,浏览器开始解释代码,会将window分两个模块:存储模块、执行模块。
存储模块,找到所有的varfunction关键字,给这些变量添加内存地址
执行模块,代码从上到下执行,遇到变量就回去存储模块查找是否有该变量
如果有该变量,就看是否赋值,如果赋值了就是后面的值,没有赋值就是undefined
如果没找到 结果就是xxx is not defined
对于作用域可能对于部分人来说是难以理解的知识点,今天呢就列举5个知识点,只要记住这几句话,作用域对你来说就是小意思

一、“JavaScript中无块级作用域”

在Java或C#中存在块级作用域,即:大括号也是一个作用域。

public static void main (){
if(1==1){
String name = "seven";
}
System.out.println(name);
}
// 报错

在JavaScript语言中无块级作用域

function Zhoumao() {
if(1 == 1){
var name = '昼猫'
}
console.log(name);
}
Zhoumao()
// 输出:昼猫

补充:标题之所以添加双引号是因为JavaScript6中新引入了 let 关键字,用于指定变量属于块级作用域。


二、JavaScript采用函数作用域

在JavaScript中每个函数作为一个作用域,在外部无法访问内部作用域中的变量。

function Zhoumao() {
var name = '昼猫'
}
Zhoumao()
console.log(name);
// 报错:ReferenceError: name is not defined

三、JavaScript的作用域链

由于JavaScript中的每个函数作为一个作用域,如果出现函数嵌套函数,则就会出现作用域链。

name = '昼猫'
function out() {
var name = 'zhoumao'
function inner() {
var name = 'hello'
console.log(name);
}
inner()
}
out()

如上述代码则出现三个作用域组成的作用域链,如果出现作用域链后,那么寻找变量时候就会出现顺序,对于上述实例:

当执行console.log(name)时,寻找顺序根据作用域链从内到外的优先级寻找,如果内层没有就逐步向上找,直到window为止,window有就使用没有就is not defined。


四、JavaScript的作用域链执行前已创建

JavaScript的作用域在被执行之前已经创建,日后再去执行时只需要按照作用域链去寻找即可。

示例一:

name = 'zhoumao'
function out() {
var name = '昼猫'
function inner() {
console.log(name);
}
return inner;
}
var code = out();
code();
// 输出结果:昼猫

上述代码,在函数被调用之前作用域链已经存在:
全局作用域 -> out函数作用域 -> inner函数作用域

当执行code()时,由于其代指的是inner函数,此函数的作用域链在执行之前已经被定义为:全局作用域 -> out函数作用域 -> inner函数作用域,所以,在执行code()时,会根据已经存在的作用域链去寻找变量。

示例二:

name = 'zhoumao'
function out() {
var name = '昼猫'
function inner() {
console.log(name);
}
name = 'hello'
return inner;
}
var code = out();
code();
// 输出结果:hello

上述代码和示例一的目的相同,也是强调在函数被调用之前作用域链已经存在:

全局作用域 -> out函数作用域 -> inner函数作用域

不同的是,在执行 var code = out()时,out作用域中的name变量的值已经由昼猫改变为hello,所以之后再执行code()时,就只能找到hello

示例三:

name = '昼猫';
function one(){
console.log(name);
}
function two(){
var name = "zhoumao";
return one;
}
var code = two();
code();
// 输出结果: 昼猫

上述代码,在函数被执行之前已经创建了两条作用域链:

全局作用域 -> one函数作用域
全局作用域 -> two函数作用域

当执行code()时,code代指的one函数,而one函数的作用域链已经存在:全局作用域 -> one函数作用域,所以,执行时会根据已经存在的作用域链去寻找。


五、声明提前

在JavaScript中如果不创建变量,直接去使用,则报错:

console.log(name)
// 报错:ReferenceError: name is not defined

如果只创建不赋值,则返回undefined

var name;
console.log(name)
// 输出:undefined

在函数中如果这么写:

function Test() {
console.log(name);
var name = '昼猫';
}
Test()
// 输出:undefined

在上述代码中不报错而是输出undefined,是因为JavaScript的函数在被执行之前,会将其中的变量全部声明,而不赋值。所以,相当于上述示例中,函数在“预编译”时,已经执行了var name;所以上述代码中输出的是undefined

类似这种写法

function Test() {
var name
console.log(name);
name = '昼猫';
}
Test()

文章来自公众号 昼猫笔记

昼猫笔记 JavaScript -- 作用域技巧!!的更多相关文章

  1. 昼猫笔记 JavaScript -- 异步执行 | 定时器真的定时执行?

      本篇主要内容:异步.定时器引发的思考 预计阅读时间:8分钟 了解 我们都知道在js中定时器有两种  setInterval()  . setTimeout()   setInterval() :按 ...

  2. 昼猫笔记 JavaScript -- 面向对象(I)

    本文内容搬运自公众号 原文链接 本文主要内容:面向对象 预计阅读时间:6分钟 面向对象的方式 单例模式(字面量定义) var obj = {} 类的实例 var obj = new Object() ...

  3. 昼猫笔记 JavaScript -- 闭包

      本次主要内容是 闭包 阅读时间: 约 3分钟 记得点个赞支持支持我哦 初步了解 先看下代码,输出结果是多少? function fn1 () { var a = 2 function fn2 () ...

  4. 昼猫笔记 从此告别复杂代码--JavaScript

    昼猫笔记--给你带来不一样的笔记 不止是笔记 更多的是思考 大家好,我是一只来自喵星的前端初学者,由于我们喵星人科技较为落后,昼猫从今天开始带着使命来到地球学习前端知识. 从今天开始,猫猫我就从Jav ...

  5. 昼猫笔记 -- 面向对象(II) - 继承

    继承 由于js不像java那样是真正面向对象的语言,js是基于对象的,它没有类的概念. 所以,要想实现继承,可以用js的原型prototype机制或者用apply和call方法去实现,还有就是js可以 ...

  6. 昼猫笔记--什么是DOM

    昼猫笔记--给你带来不一样的笔记 不止是笔记 更多的是思考 Hello,大家好,昼猫,今天来加深下DOM 什么DOM呢?它的全称叫 Document Object Model 通过全称可以知道它是 文 ...

  7. JavaScript作用域学习笔记

    重点知识点摘要 一 函数对象和其它对象一样,拥有可以通过代码访问的属性和一系列仅供JavaScript引擎访问的内部属性 其中一个内部属性是[[Scope]],由ECMA-262标准第三版定义,该内部 ...

  8. javascript作用域链学习笔记

    作用域链 "JavaScript中的函数运行在它们被定义的作用域里,而不是它们被执行的作用域里." --权威指南 在JavaScript中,一切皆对象,包括函数.函数对象和其它对象 ...

  9. ife task0003学习笔记(一):JavaScript作用域

    在学习JavaScript作用域概念之前,首先要明白几个概念:执行环境.变量对象.作用域链. 一.JavaScript执行环境(execution context): 在<Professiona ...

随机推荐

  1. MySQL 以及 Python 实现排名窗体函数

    大部分数据库都提供了窗体函数.比方RANK,ROW_NUMBER等等. MySQL 这方面没有直接提供.可是能够变相的实现.我曾经写了row_number 的实现,今天有时间把 rank 的实现贴出来 ...

  2. UVA 1415 - Gauss Prime(数论,高斯素数拓展)

    UVA 1415 - Gauss Prime 题目链接 题意:给定a + bi,推断是否是高斯素数,i = sqrt(-2). 思路:普通的高斯素数i = sqrt(-1),推断方法为: 1.假设a或 ...

  3. 高速掌握Lua 5.3 —— I/O库 (1)

    Q:什么是"Simple Model"? A:全部的文件操作都基于一个默认的输入文件和一个默认的输出文件.这就意味着同一时间对于输入和输出来说,仅仅可操作一个文件(默认的文件). ...

  4. Linux 经常使用快捷键

    桌面下: Alt+F5   取消最大化窗体 Alt+F9   最小化窗体  Alt+F10  最大化窗体  Alt+空格 打开窗体的控制菜单 (点击窗体左上角图标出现的菜单)     ctl+r   ...

  5. POJ 3256 Cow Picnic

    Cow Picnic Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 4928   Accepted: 2019 Descri ...

  6. jquery操作select的各种方法

    在工作中,有时候会遇到给select组件添加一些事件,前两天发表了一篇文章,<用jquery给select加选中事件>大致阐述了简单的jq操作select的方法,但是为了详细的介绍一下se ...

  7. SSRS 报表 递归列表

    SSRS 报表 递归列表 .需要数据集合中两个必备字段 ID PID 1.添加数据集合,在图上标记的地方点击右键添加数据集合,根据需求自己编写 2.点击插入选项卡 中的 矩阵  有两种方式 一种是 矩 ...

  8. PostgreSQL Replication之第四章 设置异步复制(8)

    4.8 处理时间线 时间线是一个您必须要知道的一个重要的概念,尤其是当您规划一个大型的设置的时候. 那么,什么是时间线呢?事实上,它是XLOG的一个分支.正常情况下,刚设置的一个数据库实例使用的时间线 ...

  9. PostgreSQL 批量生成数据

    create table user_info(userid int,name text,birthday date,crt_time timestamp without time zone,); in ...

  10. decision tree 决策树(一)

    一 决策树 原理:分类决策树模型是一种描述对实例进行分类的树形结构.决策树由结点(node)和有向边(directed edge)组成.结点有两种类型:内部结点(internal node)和叶结点( ...