1. 外层变量在内部可以找到,反之找不到

以下看个案例:

var a=10;
function aaa(){
alert(a);
} function bbb(){
var a=20;
aaa();
} bbb();

结果是?

aaa()和外层的a=10处于同一个变量作用域。所以只能查找10.

如果把a=10去掉呢?结果是报错了。

还是变量作用域的问题,aaa的环境在全局环境下,不可能找到bbb里面的a=20.

2. var的问题:不var就是全局变量

var不写也是可以的。但是不写可能产生一些问题——变量会变成一个全局变量!

由此衍生出这样的问题:

function aaa(){
var a=b=10;
}
aaa();
alert(a);
alert(b);

回想问题,b变成了一个全局变量,a是一个局部变量。

所以alert(a)会报错。

去掉alert(a),aaa()运行的结果是产生了一个全局变量b和一个局部变量a。所以alert(b)的结果是10。

3.变量查找是就近原则,寻找var定义的变量

var a=10;
function aaa(){
alert(a);
var a=20;
} aaa();

结果是什么?

既不是10也不是20。是undefined

当就近未找到,就会查找外层。一层一层知道直到找到为止。把结果简化一下吧:

var a=10;
function aaa(){
var a=20;
alert(a);
} aaa();

结果是20。这很符合常识的推断。

var a=10;
function aaa(){
a=20;
alert(a);
} aaa();

弹出a也是20,但是运行过程需要注意:最开始查找10,接着查找内部,发现a被修改为20.——本质是调用了外层的a。

好了。现在回到本节最初的例子,aaa调用alert的时候,查找的是外层的a,这个进程遇到var a=20这一步时,由于函数内部预解析,查找的作用域就变了。开始查找函数内部的a,但是var a=20这一步放在了后面,所以结果是undefined。

实际上代码变成了这样:

var a=10;
function aaa(){
var a;
alert(a);
a=20;
} aaa();

所以所有变量在定义时必须放最前面、

4.结合分析

var a=10;
function aaa(){
bbb();
alert(a);
function bbb(){
var a=20;
}
} aaa();

的结果是什么?

弹出结果是10。因为a是局部中的局部,说白了找不着。

那如果这么写呢?

var a=10;
function aaa(){
bbb();
alert(a);
function bbb(){
return a=20;
}
} aaa();

好了。由于函数的预解析作用,导致bbb执行修改了全局变量a。所以弹出结果是20.同理,如果我把全局变量var a=10删去。结果还是20。因为bbb又创造了一个全局变量a。

5. 参数跟局部变量同名,优先级是等同的

var a=10;
function aaa(a){
alert(a);
} aaa(a);

这个简单的例子中弹出的是10。

如果把参数改成b,

var a=10;
function aaa(b){
alert(a);
} aaa();

结果还是10.

那么这个a究竟是参数呢,还是全局变量a?

答案是,参数名和全局变量名一样时,走的是参数,不一样时,走的是全局变量。

比如:

var a=10;
function aaa(a){
a+=3;
}
aaa(a);
alert(a);//结果是10

基本数据类型不存在引用关系,里面的a就是一个局部变量

看个纠结一点点的吧:

var a=5;
var b=a;
b+=3; alert(a)

a是5。这是常识。

基本类型的赋值是不存在引用关系的。但如果我想让a和b存在引用关系,应该怎么做?

var a={
a:5
}
var b=a;
b.a+=3;
console.log(a);//{a:8}

如果不存在依存关系——

var a={
a:5
}
var b={
a:a.a
};
b.a+=3;
console.log(a);

另外一个简单例子是数组。

假设这样

var a=[1,2,3];
var b=a; b.push(4);
console.log(a);//[1,2,3,4]

所以真的要小心了。如果你不想存在引用关系,应该

var b=a.slice()

复合对象应用关系还可以衍生出这样的操作:

var a=[1,2,3];

function aaa(a){
a.push(4);
}
aaa(a);
console.log(a);//[1,2,3,4]

真是没有做不到,只有想不到,这个a又是什么鬼?不是局部变量吗?

a是确实是局部变量没错,但是它引用了外部的数组a。对这个局部变量的操作必将导致外部a的改变!

下面的代码或许让人更清醒些——

var a=[1,2,3];

function aaa(a){
a=[1,2,3,4]
}
aaa(a);
console.log(a);//[1,2,3,4]

因为参数a没有引用外部的a,所以怎么操作都跟外部的a没半毛钱关系!

javascript练习题(2):变量作用域的更多相关文章

  1. javascript中的变量作用域以及变量提升

    在javascript中, 理解变量的作用域以及变量提升是非常有必要的.这个看起来是否很简单,但其实并不是你想的那样,还要一些重要的细节你需要理解. 变量作用域 “一个变量的作用域表示这个变量存在的上 ...

  2. JavaScript基础——理解变量作用域

    一旦你开始在JavaScript应用程序中添加条件.函数和循环,就需要理解变量作用域.变量作用域规定了如何确定正在执行的代码行上的一个特定变量名的值. JavaScript允许你既定义全局版本又定义局 ...

  3. javascript中的变量作用域以及变量提升详细介绍

    在javascript中, 理解变量的作用域以及变量提升是非常有必要的.这个看起来是否很简单,但其实并不是你想的那样,还要一些重要的细节你需要理解变量作用域 “一个变量的作用域表示这个变量存在的上下文 ...

  4. 【JavaScript 从零开始】变量作用域

    变量作用域 一个变量的作用域(scope)是程序源代码中定义这个变量的区域. 全局变量拥有全局作用域,在JavaScript代码中的任何地方都是有定义的.然而在函数内声明的变量只是函数体内有定义. 他 ...

  5. JavaScript学习系列2一JavaScript中的变量作用域

    在写这篇文章之前,再次提醒一下 JavaScript 是大小写敏感的语言 // 'test', 'Test', 'TeSt' , 'TEST' 是4个不同的变量名 JavaScript中的变量,最重要 ...

  6. JavaScript要点 (一) 变量-作用域

    JavaScript 作用域 作用域—可访问变量的集合. 全局变量或者函数可以覆盖window对象的变量或者函数: 局部变量和window对象可以覆盖全局变量和函数. JavaScript 作用域 在 ...

  7. javascript中的变量作用域

    在网上看了一道js面试题 <script type="text/javascript"> var tt = 'aa'; function test() { alert( ...

  8. javascript痛点之一变量作用域

    1.用var声明的变量是有作用域的,比如我们在函数中用var声明一个变量 1 'use strict'; 2 function num(){ 3 //用var声明一个变量num1 4 var num1 ...

  9. 【Javascript系列】变量作用域

    问题描述 本篇文章主要讲解javascript变量及其作用域. 1   内容区 在js中,变量大致可分为全局变量(全局作用域)和局部变量(局部作用域): 用关键字var定义变量(全局变量,可以省略va ...

  10. JavaScript练习题 全局变量 局部变量 作用域

    前沿:大家好~我是阿飞~本次 任何简单的事情都可以复杂化,本次让我们来做下搞事情的练习题吧 例题1: var a = 1; function fn1(){ var a = 2; alert(a); / ...

随机推荐

  1. AutoArchive settings explained

    AutoArchive settings explained Applies To: Outlook 2010 More... Less AutoArchive helps manage the sp ...

  2. delphi inifile 支持 utf8

    1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 3 ...

  3. spring 实现定时任务

    spring实现定时任务超级简单.比使用quartz简单,比使用timer强大.如下是一个简单的springboot任务,启用了定时任务 @SpringBootApplication@Componen ...

  4. 【我的Android进阶之旅】Android调用JNI出错 java.lang.UnsatisfiedLinkError: No implementation found for的解决方法

    错误描述 今天使用第三方的so库时候,调用JNI方法时出现了错误.报错如下所示: 11-01 16:39:20.979 4669-4669/com.netease.xtc.cloudmusic E/a ...

  5. 通过文件对照工具Merge数据库

    项目分成线下开发版.线上測试版.线上生产版,因此相应有三个数据库. 对于一些静态数据.经常须要同步.改动了线下的开发版本号,同一时候也须要更新线上的測试版和线上生产版数据库,有时候线上的一些数据库改动 ...

  6. 【Java编程】写入、读取、遍历Properties文件

    在Java开发中通常我们会存储配置參数信息到属性文件.这种属性文件能够是拥有键值对的属性文件,也能够是XML文件.关于XML文件的操作,请參考博文[Java编程]DOM XML Parser 解析.遍 ...

  7. codeigniter 中使用 phpexcel

    参考:Easily integrate/load PHPExcel into CodeIgniter Framework In order to get PHPExcel working with C ...

  8. pkg-config用法和gcc cflags

    pkg-config程序是干什么用的?简单的说就是向用户向程序提供相应库的路径.版本号等信息的程序. 譬如说我们运行以下命令:pkg-config  查看gcc的CFLAGS参数 $pkg-confi ...

  9. 字符数组(char)和字符串(string)的转换

    #include<iostream>#include<string>using namespace std;void main(){ string LyuS = "W ...

  10. json字符串转化为json对象and 对象转化为 json字符串

    第一种方法: var data =evel('('+jsonstr+')') 解析:  这种方法是常用的方法, 即动态执行 javascript代码 在堆中存放数据. 存在安全问题. 第二种方法:   ...