昼猫笔记 JavaScript -- 作用域技巧!!
简单理解
var zm = function (x) {
var code = 'bb'
return code
};
学过js的老哥们都知道,当这样简单的一个函数进入浏览器,浏览器开始解释代码,会将window分两个模块:存储模块、执行模块。
存储模块,找到所有的var和function关键字,给这些变量添加内存地址
执行模块,代码从上到下执行,遇到变量就回去存储模块查找是否有该变量
如果有该变量,就看是否赋值,如果赋值了就是后面的值,没有赋值就是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 -- 作用域技巧!!的更多相关文章
- 昼猫笔记 JavaScript -- 异步执行 | 定时器真的定时执行?
本篇主要内容:异步.定时器引发的思考 预计阅读时间:8分钟 了解 我们都知道在js中定时器有两种 setInterval() . setTimeout() setInterval() :按 ...
- 昼猫笔记 JavaScript -- 面向对象(I)
本文内容搬运自公众号 原文链接 本文主要内容:面向对象 预计阅读时间:6分钟 面向对象的方式 单例模式(字面量定义) var obj = {} 类的实例 var obj = new Object() ...
- 昼猫笔记 JavaScript -- 闭包
本次主要内容是 闭包 阅读时间: 约 3分钟 记得点个赞支持支持我哦 初步了解 先看下代码,输出结果是多少? function fn1 () { var a = 2 function fn2 () ...
- 昼猫笔记 从此告别复杂代码--JavaScript
昼猫笔记--给你带来不一样的笔记 不止是笔记 更多的是思考 大家好,我是一只来自喵星的前端初学者,由于我们喵星人科技较为落后,昼猫从今天开始带着使命来到地球学习前端知识. 从今天开始,猫猫我就从Jav ...
- 昼猫笔记 -- 面向对象(II) - 继承
继承 由于js不像java那样是真正面向对象的语言,js是基于对象的,它没有类的概念. 所以,要想实现继承,可以用js的原型prototype机制或者用apply和call方法去实现,还有就是js可以 ...
- 昼猫笔记--什么是DOM
昼猫笔记--给你带来不一样的笔记 不止是笔记 更多的是思考 Hello,大家好,昼猫,今天来加深下DOM 什么DOM呢?它的全称叫 Document Object Model 通过全称可以知道它是 文 ...
- JavaScript作用域学习笔记
重点知识点摘要 一 函数对象和其它对象一样,拥有可以通过代码访问的属性和一系列仅供JavaScript引擎访问的内部属性 其中一个内部属性是[[Scope]],由ECMA-262标准第三版定义,该内部 ...
- javascript作用域链学习笔记
作用域链 "JavaScript中的函数运行在它们被定义的作用域里,而不是它们被执行的作用域里." --权威指南 在JavaScript中,一切皆对象,包括函数.函数对象和其它对象 ...
- ife task0003学习笔记(一):JavaScript作用域
在学习JavaScript作用域概念之前,首先要明白几个概念:执行环境.变量对象.作用域链. 一.JavaScript执行环境(execution context): 在<Professiona ...
随机推荐
- MySQL 以及 Python 实现排名窗体函数
大部分数据库都提供了窗体函数.比方RANK,ROW_NUMBER等等. MySQL 这方面没有直接提供.可是能够变相的实现.我曾经写了row_number 的实现,今天有时间把 rank 的实现贴出来 ...
- UVA 1415 - Gauss Prime(数论,高斯素数拓展)
UVA 1415 - Gauss Prime 题目链接 题意:给定a + bi,推断是否是高斯素数,i = sqrt(-2). 思路:普通的高斯素数i = sqrt(-1),推断方法为: 1.假设a或 ...
- 高速掌握Lua 5.3 —— I/O库 (1)
Q:什么是"Simple Model"? A:全部的文件操作都基于一个默认的输入文件和一个默认的输出文件.这就意味着同一时间对于输入和输出来说,仅仅可操作一个文件(默认的文件). ...
- Linux 经常使用快捷键
桌面下: Alt+F5 取消最大化窗体 Alt+F9 最小化窗体 Alt+F10 最大化窗体 Alt+空格 打开窗体的控制菜单 (点击窗体左上角图标出现的菜单) ctl+r ...
- POJ 3256 Cow Picnic
Cow Picnic Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 4928 Accepted: 2019 Descri ...
- jquery操作select的各种方法
在工作中,有时候会遇到给select组件添加一些事件,前两天发表了一篇文章,<用jquery给select加选中事件>大致阐述了简单的jq操作select的方法,但是为了详细的介绍一下se ...
- SSRS 报表 递归列表
SSRS 报表 递归列表 .需要数据集合中两个必备字段 ID PID 1.添加数据集合,在图上标记的地方点击右键添加数据集合,根据需求自己编写 2.点击插入选项卡 中的 矩阵 有两种方式 一种是 矩 ...
- PostgreSQL Replication之第四章 设置异步复制(8)
4.8 处理时间线 时间线是一个您必须要知道的一个重要的概念,尤其是当您规划一个大型的设置的时候. 那么,什么是时间线呢?事实上,它是XLOG的一个分支.正常情况下,刚设置的一个数据库实例使用的时间线 ...
- PostgreSQL 批量生成数据
create table user_info(userid int,name text,birthday date,crt_time timestamp without time zone,); in ...
- decision tree 决策树(一)
一 决策树 原理:分类决策树模型是一种描述对实例进行分类的树形结构.决策树由结点(node)和有向边(directed edge)组成.结点有两种类型:内部结点(internal node)和叶结点( ...