JavaScript 变量作用域
一. 变量声明
变量用var关键字来声明,如下所示:
var num;
var a,b;
var name = "ting";
var i=0, j=1, k=2;
用var关键字申明的变量是永久的,用delete运算符删除不起作用。
var i = 2;
delete i;
console.log(i); // k = 3;
delete k;
console.log(k); //报错:ReferenceError: k is not defined.
JavaScript变量是松散类型的,可以用来保存任何数据类型。并且还可在修改变量的值时改变变量的类型。
var answer = 10;
answer = "The answer is: " + 10;
console.log(answer); //
console.log(typeof answer); //string
二. 变量作用域
变量的作用域是程序中定义这个变量的区域。函数内部声明的变量只在函数内部起作用。声明局部变量一定要使用var关键字声明。
在函数内部,局部变量作用域高于同名全局变量。
var i = 99;
function foo() {
var i = 100;
console.log(i); //100(使用局部变量)
}
foo();
console.log(i); //99(使用全局变量)
声明局部变量未使用var关键字,该变量会暴露在全局环境中,与现有的全局变量冲突。如下代码中,函数内部的变量a未使用var关键字,与全局环境中的a冲突,在全局环境中调用a。
var a = 5;
function foo() {
a = 10;
b = 15;
console.log(a); //
console.log(b); //
}
foo();
console.log(a); //
console.log(b); //
变量在未声明的情况下被初始化,会被添加到全局环境。
var add = function(a,b) {
var sum = a + b;
return sum;
}
var res = add(2,3);
console.log(res); //
console.log(sum); //ReferenceError: sum is not defined var add = function(a,b) {
sum = a + b;
return sum;
}
var res = add(2,3);
console.log(sum); //
JavaScript执行代码时,会创建一个上下文执行环境,全局环境是最外围的环境。每个函数在被调用时都会创建自己的执行环境,当函数执行完,当前执行环境被销毁。
每个执行环境都有一个与之关联的作用域链。在执行代码时,JavaScript引擎会通过搜索执行环境的作用域链来解析变量和函数名这样的标识符。 解析过程从作用域链的前端开始,向上逐级查询与给定名字匹配的标识符,一旦找到标识符,搜索过程就停止,否则继续沿作用域链向上搜索,一直搜索到全局对象。如果没有搜到,则认为该标识符未定义。标识符在作用域链中的位置越深,查找和访问它的时间越长,所以尽可能使用局部变量。
全局环境只能访问在全局环境中定义的变量和函数,不能直接访问局部环境中的任何数据。
三. 没有块级作用域
JavaScript中没有块级作用域,举个例子。如果JavaScript中有块级作用域的话,控制台输出的应该是undefined。但实际上控制台输出的是100,这是为什么呢? 原因在于:没有块级作用域,变量i没有被销毁,因此还能访问到这个变量。
for(var i=0; i<100; i++) {
//todo
}
console.log(i); //
JavaScript中存在函数作用域。当foo函数执行完毕后,函数内的变量将被销毁,因此控制台报错。
function foo() {
var bar = "fn";
}
foo();
console.log(bar); //报错:ReferenceError: bar is not defined.
函数作用域模拟块级作用域。
(function() {
for(var i=0; i<100; i++) {
//todo
}
})();
console.log(i); //报错:ReferenceError: bar is not defined.
再看一例:局部变量a在整个foo函数内部都有定义的,隐藏了全局变量a。但第一个console在变量初始化之前,所以其值为undefined。
var a = 8;
var foo = function() {
console.log(a); //undefined
var a = 5;
console.log(a); //
};
foo();
四. 未赋值的变量和未定义的变量
未赋值的变量值为undefined,而使用未定义的变量会引起错误。
var a;
console.log(a); //undefined
console.log(b); //ReferenceError: b is not defined
五. 垃圾收集(garbage collection)
JavaScript具有自动垃圾收集机制,Javascript解释器可以检测到何时程序不再使用一个对象,就把它所占用的内存释放。
时间:2014-10-22
地点:合肥
引用:http://wlog.cn/javascript/javascript-variable-scope-chain.html
http://segmentfault.com/blog/liangyi/1190000000692129
JavaScript 变量作用域的更多相关文章
- 【转】javascript变量作用域、匿名函数及闭包
下面这段话为摘抄,看到网上大多数人使用的是变量在使用的时候声明而不是在顶端声明,也可能考虑到js查找变量影响性能的问题,哪里用就在哪里声明,也很好. 在Javascript中,我们在写函数的时候往往需 ...
- 第一百零六节,JavaScript变量作用域及内存
JavaScript变量作用域及内存 学习要点: 1.变量及作用域 2.内存问题 JavaScript的变量与其他语言的变量有很大区别.JavaScript变量是松散型的(不强制类型)本质,决定了它只 ...
- JavaScript变量作用域
全部变量拥有全局作用域,局部变量拥有局部作用域(这里注意函数的参数也是局部变量) 1.在函数体内,局部变量的优先级高于同名的全局变量. 我的理解就是当你同时定义了同名的局部变量和全局变量时,函数体内返 ...
- 深入理解Javascript变量作用域
在学习JavaScript的变量作用域之前,我们应当明确几点: a.JavaScript的变量作用域是基于其特有的作用域链的. b.JavaScript没有块级作用域. c.函数中声明的变量在整个函数 ...
- JavaScript 变量作用域 详解
变量作用域要点 - 在JavaScript中没有块级作用域,只有函数作用域 - 在函数体内,局部变量的优先级高于同名的全局变量 - 在全局作用域编写代码时可以不写var语句,但声明局部变量时必须使用v ...
- JavaScript变量作用域(Variable Scope)和闭包(closure)的基础知识
在这篇文章中,我会试图讲解JavaScript变量的作用域和声明提升,以及许多隐隐藏的陷阱.为了确保我们不会碰到不可预见的问题,我们必须真正理解这些概念. 基本定义 作用范围是个“木桶”,里面装着变量 ...
- 基础系列(1)之干掉JavaScript变量作用域
今天去某顺公司面试,发现一些基础知识都不记得了,于是乎决定把js基础系列的全部梳理一遍,今天就整理下js变量作用域的相关基础知识点,配合最常遇到的笔试题阐述. 题一: var g = "a ...
- 深入浅出JavaScript变量作用域
在学习JavaScript的变量作用域之前,我们应当明确几点: JavaScript的变量作用域是基于其特有的作用域链的. JavaScript没有块级作用域. 函数中声明的变量在整个函数中都有定义. ...
- JavaScript 变量作用域和声明提升
一.变量作用域 说到这个概念,不有自主的想到this,scope 这两个关键字. JavaScript的this总是指向一个明确的对象,这个对象是在执行的时候动态绑定的.通俗的说就是谁调用我,我的th ...
随机推荐
- FtpClient.storeFile返回false解决方法
在确定路径和文件名没有中文的情况下添加以下代码 ftp.setFileTransferMode(ftp.BINARY_FILE_TYPE); ftp.enterLocalPassiveMode();/ ...
- Android中Context详解
大家好, 今天给大家介绍下我们在应用开发中最熟悉而陌生的朋友-----Context类 ,说它熟悉,是应为我们在开发中时刻的在与它打交道,例如:Service.BroadcastReceiver.A ...
- OpenCV学习笔记(一)——OpenCV安装
1.无脑安装以下安装文件 cn_visual_studio_2010_ultimate_x86_dvd_532347.iso 2.测试Hello OpenCV 文件→新建→项目 win32应用程序→下 ...
- MemCache中的内存管理详解
MC的内存管理机制 1.内存的碎片化 当我们使用C语言或者其他语言进行malloc(申请内存),free(释放内存)等类似的命令操作内存的时候, 在不断的申请和释放的过程中,形成了一些很小的内存片段, ...
- MySQL wamp密码修改
WAMP安装好后,mysql密码是为空的,那么要如何修改呢?其实很简单,通过几条指令就行了,下面我就一步步来操作. 首先,通过WAMP打开mysql控制台. 提示输入密码,因为现在是空,所以直接按回车 ...
- `UnityEditor.EditorUtility' does not contain a definition for `GetMiniThumbnail'
I got the following errors with Untiy 4.0f7error CS0117: `UnityEditor.EditorUtility' does not contai ...
- 【OS】分页和分段
在网上找到了一个比较形象的比喻 打个比方,比如说你去听课,带了一个纸质笔记本做笔记.笔记本有100张纸,课程有语文.数学.英语三门,对于这个笔记本的使用,为了便于以后复习方便,你可以有两种选择. ...
- [ubuntu]用ubuntu开发的日子----win7 ubuntu双系统
小子终于忍不了win7某些蛋疼的设定,看群里好多大牛推荐mac,但资金紧张,只好推而求其次使用ubuntu,但是由于公司工作环境是windows,所以还必须保留windows系统,一次决定双系统. 下 ...
- [HTML/HTML5]6 使用图像
6.1 将图像作为Web页面的前景元素 使用img元素,就可以在Web页面中添加图像.img是image的缩写.只需在img元素中添加src(source的缩写)属性,并为该属性设置相应的值,就可以 ...
- Python中文乱码
1,注意:请使用智慧型浏览器 "CHROME" 配合理解和运作本文中提到的程序. 2,提示:谷歌的CHROME浏览器是迄今为止最智慧的浏览器,没有之一,只有第一. 3,谷歌的CHR ...