论JavaScript的作用域
一直以来本人认为想深入了解一门语言,不光是让自己变成撸sir,更需要时间的锤炼。能经得起时间考验的东西更值得拥有。学习和使用Javascript一晃都7年了,最近才感觉自己对他才有顿悟,不知道是否来得有点迟。本文归纳了我对 JS中作用的理解,希望得学习有所帮助。
特别说明:这是从另一个侧面(函数域的覆盖范围)来解释和说明执行上下文。
一、作用域的理论理解
从入门Javascript时,无论是学校老师,还是你工作的老司机,都会很认真的考虑你,Js中有一个全局作用域,然后他包含很多的子域(如:由function、object创建作用域的),子域是可以很轻松的访问父级域中的任何对象。这种表述你是否感觉很是空洞,难以理解。今本人就用浩瀚无边的地域来举个粟子:
通过上面这个Image,让我一步一步带你领略作用域的魅力。
1. 首先,地球:我们大家都很熟悉,他包含着你,我,以及世界上所有的国家(这儿就不论宇宙,否则会Hold不住),这肯定就是Js中的全局作用域了。地球包含天空、空气,正如Js中全局作用包含包含Number、String、Function、Object、Boolean、Regex等对象。
2. 再者:中国、美国、朝鲜是三个平等的子作用域,他们都有利用地球 这个全局作用域的权利,如发射卫星。但美国再看不懂朝鲜,也不可能到朝鲜去干什么事情的。如Js的子作用域中可以任意使用Number、String等对象,但是没办法直接调用一个作用域平级的对象的方法。
3. 再次:你做为一个中国公民,可以依法使用中国内部的资源(只不过要么交钱买,要么就是拿了税的),如果你想买点美国的的东西,那就需要new一个什么宝、或者什么东的来做中国与美国之间的引用,然后您懂的。如Js中需要访问一个平级作用域的内容时,那你就需要拿到被访问的引用。
4. 最后:bind,apply,call,可以这样理解,bind就是在正规海购实体商场东西(预先可以看到实体的物品),而apply和call则是代购(只能视频看)。bind改变一个作用但此函数不会立刻执行,而apply和call则会立刻执行。
二、示例代码说明作用域
function Card(){
this.name = 'name';
}
Card.prototype.getName =function(callback){
callback();
return this.name;
} function PostCard(card){
this.name = "postCard"
this.card =card;
}
PostCard.prototype.getName =function(){
var _name = this.card.getName(function(){
console.log("callback: " + this.name);
});
console.log(_name + " " + this.name);
} var card = new Card();
var postCard = new PostCard(card);
postCard.getName();
1. 上述代码Card和PostCard就像第一部分提及的中国、美国一样,他们都有一个共同的父级作用域,就是window(地球)
2. Card和PostCard是平级作用域,如PostCard的getName方法想访问Card的getName方法,就需要拿到Card的引用,如PostCard构造函数传入的card实例就是为了完成这个事情。
3. Card的getName接受一个callback的高阶函数(这又是另外一个话题,函数式编程了),这里只要记住传入就是一个类型为function的形参就行了。然后在Card的getName方法中执行了callback,但这里你要注意,执行callback时没有通过任何方式或手段来指定他的作用域,所以callback执行作用域为window(地球)。
4.输出结果:
callback:
name postname
三、示例代码说明bind\apply\call
function Direction(){
this.direction = "mid";
}
Direction.prototype.set =function(val){
this.direction = val;
} var direction = new Direction();
var dir02 = new Direction();
direction.set.call(direction, "right");
direction.set.apply(dir02, ["bottom"]); console.log(direction.direction);
console.log(dir02.direction); //bind
var tmp = direction.set.bind(dir02);
tmp("bind after");
console.log('..........................');
console.log(direction.direction);
console.log(dir02.direction);
本示例代码提供了一个Direction函数,然后对this(当前实例)绑定了direction属性,且在Direction原型上绑定了set方法(用于修改this上的direction).
1. 通过 new 创建了Direction的两个实例direction,dir02
2. 通过call和apply改变作用域的指向,进行set方法的执行,call指向了direction实例,而apply指向了dir02的实例。这也展示了call与apply的区别,就是参数传送用一个列表,一个是数组。
3. 通过输出结果大家会发现,虽然都是调用direction实例的set方法,但改变的this.direction却是对应的实例属性。
4. 通过bind方法把direction实例的set方法与dir02相绑定,然后执行这个绑定,会发现改变是仍然是dir02实例的direction属性。
5. 输出结果
right bottom
scope
..........................
right
bind after
四、作用域的总结
1. 只有一个全局作用域,但有无数个子作用域。
2. 作用域的创建与执行:
2.1 创建阶段[函数被调用,但内部代码还没开始执行]
2.2 创建 作用域链
2.3 创建变量 函数 以及参数
2.4 决定this的值(也就是作用域,或者是执行上下文)
2.5 代码执行[赋值、寻找函数引用以及解释/执行代码]
论JavaScript的作用域的更多相关文章
- 浅谈JavaScript的作用域
前段时间学了下JavaScript作用域,这个东西在JavaScript非常重要,也是JavaScript很基础的东西,正如少林里面基础武功,有了基础,才能学绝世武功. 作用域的作用是啥?一套设计良好 ...
- JavaScript变量作用域
全部变量拥有全局作用域,局部变量拥有局部作用域(这里注意函数的参数也是局部变量) 1.在函数体内,局部变量的优先级高于同名的全局变量. 我的理解就是当你同时定义了同名的局部变量和全局变量时,函数体内返 ...
- 初探JavaScript(四)——作用域链和声明提前
前言:最近恰逢毕业季,千千万万的学生党开始步入社会,告别象牙塔似的学校生活.往往在人生的各个拐点的时候,情感丰富,感触颇深,各种对过去的美好的总结,对未来的展望.与此同时,也让诸多的老“园”工看完这些 ...
- JavaScript的作用域与作用域链
作用域 作用域就是变量与函数的可访问范围,即作用域控制着变量与函数的可见性和生命周期.可以说,变量和函数在什么时候可以用,什么时候被摧毁,这都与作用域有关. JavaScript中,变量的作用域有全局 ...
- JavaScript的作用域和提升机制
JavaScript的作用域和提升机制 你知道下面的JavaScript代码执行时会输出什么吗? 1 2 3 4 5 6 7 8 var foo = 1; function bar() { i ...
- Javascript的作用域、作用域链以及闭包
一.javascript中的作用域 ①全局变量-函数体外部进行声明 ②局部变量-函数体内部进行声明 1)函数级作用域 javascript语言中局部变量不同于C#.Java等高级语言,在这些高级语言内 ...
- 深入理解Javascript变量作用域
在学习JavaScript的变量作用域之前,我们应当明确几点: a.JavaScript的变量作用域是基于其特有的作用域链的. b.JavaScript没有块级作用域. c.函数中声明的变量在整个函数 ...
- 【转】javascript变量作用域、匿名函数及闭包
下面这段话为摘抄,看到网上大多数人使用的是变量在使用的时候声明而不是在顶端声明,也可能考虑到js查找变量影响性能的问题,哪里用就在哪里声明,也很好. 在Javascript中,我们在写函数的时候往往需 ...
- 网易JS面试题与Javascript词法作用域说明
调用对象位于作用域链的前端,局部变量(在函数内部用var声明的变量).函数参数及Arguments对象都在函数内的作用域中--这意味着它们隐藏了作用域链更上层的任何同名的属性. 2010年9月14日, ...
- 第一百零六节,JavaScript变量作用域及内存
JavaScript变量作用域及内存 学习要点: 1.变量及作用域 2.内存问题 JavaScript的变量与其他语言的变量有很大区别.JavaScript变量是松散型的(不强制类型)本质,决定了它只 ...
随机推荐
- 腾讯AI开放平台的使用
一.腾讯AI开放平台 https://ai.qq.com/ 二.腾讯AI平台支持的功能 三.签名机制 1.计算步骤 用于计算签名的参数在不同接口之间会有差异,但算法过程固定如下4个步骤. 1.将< ...
- Spark调优 数据倾斜
1. Spark数据倾斜问题 Spark中的数据倾斜问题主要指shuffle过程中出现的数据倾斜问题,是由于不同的key对应的数据量不同导致的不同task所处理的数据量不同的问题. 例如,reduce ...
- gitlab使用过程中的需求与解决
序言 在git使用过程中发现指令实在太多,就算记忆后不长用的话很快也会忘记掉,所以编写本文的初衷是为了记录在使用git指令的过程中所遇到的需求与解决方法,毕竟使用git的需求也就那么一些,范围不大,所 ...
- 了解javascript里面的 封装
// 封装 //生成实例对象的原始模式 //假如我们把一个动物看成一个对象 var cat = { //那么它有名字和颜色两个属性 name:'', color:'' }; //接下来我们根据原形对象 ...
- 2018-2019-2 20165220 《网络对抗技术》Exp1 PC平台逆向破解
实验目的 本次实践的对象是一个名为pwn1的linux可执行文件. 该程序正常执行流程是:main调用foo函数,foo函数会简单回显任何用户输入的字符串. 该程序同时包含另一个代码片段,getShe ...
- 音频相关基本概念,音频处理及编解码基本框架和原理以及音、重采样、3A等音频处理(了解概念为主)
视频笔记:音频专业级分析软件(Cooledit) 音质定义以语音带宽来区分,采样率越高,带宽越大,则保真度越高,音质越好.窄带(8khz采样),宽带(16khz采样),CD音质(44.1khz采样) ...
- 短网址API
http://tao.tf/open/ API简介 API允许第三方自由调用URL缩短,基于text/json/jsonp/js模式,支持post.get提交. 支持缩短网址: 淘宝网(*.taoba ...
- windows与linux多线程对比
一.创建线程 1>windows HANDLE aThread[MAX_THREAD]; 函数原型: HANDLE WINAPI CreateThread( _In_opt_ LPSECUR ...
- python 多线程 及多线程通信,互斥锁,线程池
1.简单的多线程例子 import threading,timedef b_fun(i): print "____________b_fun start" time.sleep(7 ...
- 32位二进制IP地址与十进制IP地址互相转换
代码: import java.util.List; import java.util.ArrayList; import java.util.Scanner; public class Transf ...