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. 巨蟒python全栈开发-第5天 字典&集合

    今日大纲: 1.什么是字典 字典是以key:value的形式来保存数据,用{}表示. 存储的是key:value 2.字典的增删改查(重点) (1) 添加 dic[新key] = 值 setdefau ...

  2. GO语言中使用OpenCV

    GO语言中使用OpenCV - OpenCV China :图像处理,计算机视觉库,Image Processing, Computer Vision http://wiki.opencv.org.c ...

  3. Java 之单例设计模式

    设计模式: 对问题行之有效的解决方式, 其实它是一种思想. 单例设计模式 解决的问题:就是可以保证一个类在内存中的对象唯一性. 即单个实例. 比如对于A 和 B 两个程序使用同一个配置信息对象时, A ...

  4. redis数据库设计(转)

    原文:http://segmentfault.com/q/1010000000316112 redis是什么 redis就是一个存储key-value键值对的仓库,如何使用redis在于如何理解你需要 ...

  5. 深入Redis内部-Redis 源码讲解(转)

    Redis作为 NoSQL 数据库的杰出代表,一直广受关注,其轻量级的敏捷架构,向来有存储中的瑞士军刀之称.下面推荐的一篇文章,从源码的角度讲解了Redis 的整个工作流程,是了解 Redis 流程的 ...

  6. Frequent Values-线段树求解出现最多的数

    Frequent Values(poj 3368) 注意:以下答案为离线作答结果,并非能通过poj,若要通过poj,需要修改函数接口,因为以下程序接受半封闭区间(s,e],同时还需要修改输入数据的顺序 ...

  7. HDU3552(贪心)

    题目是将一系列点对(a,b)分成两个集合.使得A集合的最大a+B集合的最大数b得和最小. 思路:http://blog.csdn.net/dgq8211/article/details/7748078 ...

  8. mysql 建立表之间关系 练习 2

    创建数据库db6 create database db6 charset=utf8; user db6; # 创建班级表 mysql) not null unique); Query OK, rows ...

  9. exp导出一个表中符合查询条件的数据

    原文地址:exp导出一个表中符合查询条件的数据 作者:charsi 导出一个表中的部分数据,使用QUERY参数,如下导出select * from test where object_id>50 ...

  10. while循环中的break、continue和else

    break:直接结束当前循环然后跳到下面的语句.break之后在循环外continue:结束本次循环,跳到下次循环.continue之后依然还在循环内else:这是while循环所特有,当循环结束之后 ...