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. submit按钮修改宽高的坑

    近些天对h5非常感兴趣,边工作边学习,虽然比较累,但过得很踏实.每天都要学习一点东西,这样才能对得起自己.好了,废话不多说,进入今天的主题. 今天遇到了一个非常有趣的东西,就是在修改submit按钮的 ...

  2. 第00章—IDEA

    spring boot 系列学习记录:http://www.cnblogs.com/jinxiaohang/p/8111057.html 码云源码地址:https://gitee.com/jinxia ...

  3. continue、return和break的区别

     1.return 语句的作用       (1) return 从当前的方法中退出,返回到该调用的方法的语句处,继续执行.       (2) return 返回一个值给调用该方法的语句,返回值的数 ...

  4. window安装redis数据库

    1.下载安装包 1.百度网盘链接:https://pan.baidu.com/s/1MrAK5Suc1xpzkbp1WQbP0A 提取码:u9uq 2.GitHub:https://github.co ...

  5. sys模块 logging模块 序列化模块

    一 :sys模块 sys.argv 命令行参数List,第一个元素是程序本身路径 sys.exit(n) 退出程序,正常退出时exit(0) sys.version 获取Python解释程序的版本信息 ...

  6. 深入ff and ffbase

    用ff 包读取一个csv 文件 >options(fftempdir = [二进制文件存放的位置]) >file_chunks <- read.csv.ffdf(file=”big_ ...

  7. 使用Stanford Parser进行句法分析

    一.句法分析 1.定义 句法分析判断输入的单词序列(一般为句子)的构成是否合乎给定的语法,并通过构造句法树来确定句子的结构以及各层次句法成分之间的关系,即确定一个句子中的哪些词构成一个短语,哪些词是动 ...

  8. decorator & generator & iterator

    装饰器(decorator): @staticmethod @classmethod 都既可以使用类名访问,也可以使用对象名访问, 但classmethod在定义时需要cls参数 生成器(genera ...

  9. zend studio设置utf8

    1. windows -> preference -> general -> workspace 2.项目右键 -> properities -> resource 3. ...

  10. 关于OAuth的state参数的作用

    引用: https://blog.csdn.net/gjb724332682/article/details/54428808 在开发 OAuth认证服务器 的时候,开发者的安全意识不高的话,很可能会 ...