按值传递--

http://www.cnblogs.com/yizihan/p/4339814.html

// JS变量是松散型的(不强制类型)本质,决定了它只是在特定时间用于保存特定值的一个名字而已;

// 由于不存在定义某个变量必须要保存何种数据类型值的规则,变量的值及其数据类型可以在脚本的生命周期内改变;

一 变量及作用域

1.基本类型和引用类型

1 // JS变量包含两种不同的数据类型的值:基本类型值和引用类型值;
2
3 // 1.基本类型值:保存在栈内存中的简单数据段;即这种值完全保存在内存中的一个位置;
4 // 基本类型值包含:Undefined|Null|Boolean|Number|String;
5 // 这些类型在内存中占有固定大小的空间;它们的值保存在栈空间,我们按值来访问;
6
7 // 2.引用类型值:保存在堆内存中的对象(可能由多个值构成),即变量中保存的实际上只是一个指针,这个指针指向内存中的另一个位置,该位置保存对象;
8 // 引用类型的值的大小不固定,因此不能保存在栈内存,必须保存在堆内存中;但可以将引用类型的值的内存地址保存在栈内存中;
9 // 当查询引用类型的变量时,先从栈内存中读取内存地址,然后通过地址找到堆内存中的值;=>按引用访问;

2.动态属性

1 // 定义基本类型值和引用类型值的方式相似:创建一个变量并为该变量赋值;
2 // 但当这个值保存到变量中以后,对不同类型值可以执行的操作则不一样;
3     var box = new Object();                     // 创建引用类型;
4     box.name = ‘lee‘;                           // 新增一个属性;
5     console.log(box.name);                      // =>lee;
6
7     var box = ‘lee‘;                            // 创建基本类型
8     box.age = 15;                               // 给基本类型添加属性;
9     console.log(box.age);                       // =>undefined;

3.复制变量值

 1 // 在变量复制方面,基本类型和引用类型也有所不同;
 2 // 基本类型赋值的是值本身;
 3     var box = ‘lee‘;                            // 在栈内存中生成一个box‘lee‘;
 4     var box2 = box;                             // 在栈内存中再生成一个box2‘lee‘;
 5     // box和box2完全独立;两个变量分别操作时互不影响;
 6
 7 // 引用类型赋值的是地址;
 8     var box = new Object();                    // 创建一个引用类型;box在栈内存中;而Object在堆内存中;
 9     box.name = ‘lee‘;                          // 新增一个属性;
10     var box2 = box;                            // 把引用地址赋值给box2;box2在栈内存中;
11     // box2=box,因为它们指向的是同一个对象;
12     // 如果这个对象中的name属性被修改了,box.name和box2.name输出的值都会被修改掉;

4.传递参数

 1 // JS中所有函数的参数都是按值传递的,即参数不会按引用传递;
 2     function box(num){                         // 按值传递,传递的参数是基本类型;
 3         num +=10;                              // 这里的num是局部变量,全局无效;
 4         return num;
 5     }
 6     var num = 50;
 7     var result = box(num);
 8     console.log(result);                      // 60;
 9     console.log(num);                         // 50;
10
11     function box(num){
12         return num;
13     }
14     console.log(num);                         // num is not defined;
15
16     function box(obj){
17         obj.name = ‘lee‘;
18         var obj = new Object();              // 函数内部又创建了一个对象,它是局部变量;但在函数结束时被销毁了;
19         obj.name = ‘Mr‘;                     // 并没有替换掉原来的obj;
20     }
21     var p = new Object();
22     box(p);                                  // 变量p被传递到box()函数中之后就被复制给了obj;在函数内部,obj和p访问的是同一个对象;
23     console.log(p.name);                     // =>lee;
24
25     // JS函数的参数都将是局部变量;也就是说,没有按引用传递;

5.检测类型

 1 // 要检测一个变量的类型,通过typeof运算符类判断;
 2 // 多用来检测基本类型;
 3     var box = ‘lee‘;
 4     console.log(typeof box);                    // =>string;
 5
 6 // 要检测变量是什么类型的对象,通过instanceof运算符来查看;
 7     var box = [1,2,3];
 8     console.log(box instanceof Array);         // =>true;
 9     var box2 = {};
10     console.log(box2 instanceof Object);
11     var box3 = /g/;
12     console.lgo(box3 instanceof RegExp);
13     var box4 = new String(‘lee‘);
14     console.log(box4 instanceof String);      // =>true;是否是字符串对象;
15
16     var box5 = ‘string‘;
17     console.log(box5 instanceof String);      // =>false;
18     // 当使用instanceof检查基本类型的值时,它会返回false;

6.执行环境及作用域

 1 // 执行环境:定义了变量或函数有权访问的其他数据,决定了它们各自的行为;
 2 // 在Web浏览器中,全局执行环境=window对象;
 3 // 因此所有的全局变量和函数都是作为window对象的属性和方法创建的;
 4     var box = ‘blue‘;                         // 声明一个全局变量;
 5     function setBox(){
 6         console.log(box);                     // 全局变量可以在函数里访问;
 7     }
 8     setBox();                                 // 执行函数;
 9     // 全局的变量=window对象的属性;
10     // 全局的函数=window对象的方法;
11
12 // PS:当执行环境中的所有代码执行完毕后,该环境被销毁,保存在其中的所有变量和函数定义也随之销毁;
13 // 如果是在全局环境下,需要程序执行完毕,或者网页被关闭才会销毁;
14
15 // PS:每个执行环境都有一个与之关联的变量对象,就好比全局的window可以调用全局变量和全局方法一样;
16 // 局部的环境也有一个类似window的变量对象,环境中定义的所有变量和函数都保存在这个对象中;
17 // (我们无法访问这个变量对象,但解析器会处理数据时后台使用它);
18     var box = ‘blue‘;
19     function setBox(){
20         var box = ‘red‘;                      // 这里是局部变量,在当前函数体内的值是‘red‘;出了函数体就不被认知;
21         console.log(box);
22     }
23     setBox();
24     console.log(box);
25
26 // 通过传参可以替换函数体内的局部变量,但作用域仅限在函数体内这个局部环境;
27     var box = ‘blue‘;
28     function setBox(box){                     // 通过传参,将局部变量替换成了全局变量;
29         alert(box);                           // 此时box的值是外部调用时传入的参数;=>red;
30     }
31     setBox(‘red‘);
32     alert(box);
33
34 // 如果函数体内还包含着函数,只有这个内函数才可以访问外一层的函数的变量;
35 // 内部环境可以通过作用域链访问所有的外部环境,但外部环境不能访问内部环境中的任何变量和函数;
36     var box = ‘blue‘;
37     function setBox(){
38         function setColor(){
39             var b = ‘orange‘;
40             alert(box);
41             alert(b);
42         }
43         setColor();                           // setColor()的执行环境在setBox()内;
44     }
45     setBox();
46     // PS:每个函数被调用时都会创建自己的执行环境;当执行到这个函数时,函数的环境就会被推到环境栈中去执行,而执行后又在环境栈中弹出(退出),把控制权交给上一级的执行环境;
47
48     // PS:当代码在一个环境中执行时,就会形成一种叫做作用域链的东西;它的用途是保证对执行环境中有访问权限的变量和函数进行有序访问;作用域链的前端,就是执行环境的变量对象;

7.延长作用域链

 1 // 有些语句可以在作用域链的前端临时增加一个变量对象,该变量对象会在代码执行后被移除;
 2 // with语句和try-catch语句;这两个语句都会在作用域链的前端添加一个变量对象;
 3 // with语句:会将指定的对象添加到作用域链中;
 4 // catch语句:会创建一个新的变量对象,其中包含的是被抛出的错误对象的声明;
 5     function buildUrl(){
 6         var qs = ‘?debug=true‘;
 7         with(location){                       // with语句接收的是location对象,因此变量对象中就包含了location对象的所有属性和方法;
 8             var url = href+qs;                // 而这个变量对象添加到了作用域链的前端;
 9         };
10         return url;
11     }

8.没有块级作用域

 1 // 块级作用域:表示诸如if语句等有花括号封闭的代码块,所以,支持条件判断来定义变量;
 2     if(true){                                 // if语句代码块没有局部作用域;
 3         var box = ‘lee‘;                      // 变量声明将变量添加到当前的执行环境(在这里是全局环境);
 4     }
 5     alert(box);
 6
 7     for(var i=0; i<10; i++){                  // 创建的变量i即使在for循环执行结束后,也依旧会存在与循环外部的执行环境中;
 8         var box = ‘lee‘;
 9     }
10     alert(i);
11     alert(box);
12
13     function box(num1,num2){
14         var sum = num1+num2;                  // 此时sum是局部变量;如果去掉var,sum就是全局变量了;
15         return sum;
16     }
17     alert(box(10,10));
18     alert(sum);                               // sum is not defined;访问不到sum;
19     // PS:不建议不使用var就初始化变量,因为这种方法会导致各种意外发生;
20
21 // 一般确定变量都是通过搜索来确定该标识符实际代表什么;搜索方式:向上逐级查询;
22     var box = ‘blue‘;
23     function getBox(){
24         return box;                           // 此时box是全局变量;如果是var box=‘red‘,那就变成局部变量了;
25     }
26     alert(getBox());
27     // 调用getBox()时会引用变量box;
28     // 首先,搜索getBox()的变量对象,查找名为box的标识符;
29     // 然后,搜索继续下一个变量对象(全局环境的变量对象),找到了box标识符;
30 // PS:变量查询中,访问局部变量要比全局变量更快因为不需要向上搜索作用域链

二 内存问题

 1 // JS具有自动垃圾收集机制,执行环境会负责管理代码执行过程中使用的内存;它会自行管理内存分配及无用内存的回收;
 2
 3 // JS最常用的垃圾收集方式就是标记清除;垃圾收集器会在运行的时候给存储在内存中的变量加上标记;
 4 // 然后,它会去掉环境中正在使用的变量的标记,而没有被去掉标记的变量将被视为准备删除的变量;
 5 // 最后,垃圾收集器完成内存清理工作,销毁那些标记的值并回收他们所占用的内存空间;
 6
 7 // 垃圾收集器是周期性运行的,这样会导致整个程序的性能问题;
 8 // 比如IE7以前的版本,他的垃圾收集器是根据内存分配量运行的,比如256个变量就开始运行垃圾收集器,这样就不得不频繁地运行,从而降低了性能;
 9
10 // 一般来说,确保占用最少的内存可以让页面获得更好的性能;
11 // 最佳方案:一旦数据不再使用,将其值设置为null来释放引用,这个做法叫做解除引用;
12     var o = {
13         name:‘lee‘;
14     };
15     o = null;                             // 解除对象引用,等待垃圾收集器回收;

三 小结

 1 1.变量
 2 // JS变量可以保存两种类型的值:基本类型值和引用类型值;它们具有以下特点:
 3 // 1.基本类型值在内存中占据固定大小的空间,因此被保存在栈内存中;
 4 // 2.从一个变量向另一个变量复制基本类型的值,会创建这个值的一个副本;
 5 // 3.引用类型的值是对象,保存在堆内存中;
 6 // 4.包含引用类型值的变量实际上包含的并不是对象本身,而是一个指向该对象的指针;
 7 // 5.从一个变量向另一个变量复制引用类型的值,复制的其实是指针,因此两个变量最终都指向用一个对象;
 8 // 6.确定一个值是哪种基本类型可以使用typeof操作符;而确定一个值是哪种引用类型可以使用instanceof操作符;
 9
10 2.作用域
11 // 所有变量都存在于一个执行环境(作用域)中,这个执行环境决定了变量的生命周期,以及哪一部分代码可以访问其中的变量;
12 // 1.执行环境有全局执行环境和函数执行环境之分;
13 // 2.每次进入一个新执行环境,都会创建一个用于搜索变量和函数的作用域链;
14 // 3.函数的局部环境不仅有权访问函数作用域中的变量,而且有权访问其父环境,乃至全局环境;
15 // 4.变量的执行环境有助于确定应该合适释放内存;
16
17 3.内存
18 // JS自动垃圾收集机制
19 // 1.离开作用域的值将被自动标记为可以回收,因此将在垃圾收集期间被删除;
20 // 2.为了确保有效地回收内存,应该及时解除不再使用的全局对象/全局对象属性以及循环引用变量的引用;
时间: 2024-10-19 07:06:58

按值传递--的相关文章

javascript中所有函数的参数都是按值传递的

[javascript中所有函数的参数都是按值传递的] 参考:http://www.jb51.net/article/89297.htm

JS是按值传递还是按引用传递

按值传递(call by value)是最常用的求值策略:函数的形参是被调用时所传实参的副本.修改形参的值并不会影响实参. 按引用传递(call by reference)时,函数的形参接收实参的隐式引用,而不再是副本.这意味着函数形参的值如果被修改,实参也会被修改.同时两者指向相同的值. 按引用传递会使函数调用的追踪更加困难,有时也会引起一些微妙的BUG.  按值传递由于每次都需要克隆副本,对一些复杂类型,性能较低.两种传值方式都有各自的问题. JS的基本类型,是按值传递的. 1 var a

学习笔记:JavaScript传参方式———ECMAScript中所有函数的参数都是按值传递

我们把命名参数(arguments)视为局部变量,在向参数传递基本类型值时,如同基本类型变量的复制一样,传递一个副本,参数在函数内部的改变不会影响外部的基本类型值.如: 1 function add10(num){ 2 num += 10 ; 3 return num ; 4 } 5 var count = 10 ; 6 var result = add10(count); 7 alert(count);// 10 8 alert(result); //20 在向参数传递引用类型的值时,会把这个

Java中只有按值传递,没有按引用传递!(两种参数情况下都是值传递)

今天,我在一本面试书上看到了关于java的一个参数传递的问题: 写道 java中对象作为参数传递给一个方法,到底是值传递,还是引用传递? 我毫无疑问的回答:“引用传递!”,并且还觉得自己对java的这一特性很是熟悉! 结果发现,我错了! 答案是: 值传递!Java中只有按值传递,没有按引用传递! 回家后我就迫不及待地查询了这个问题,觉得自己对java这么基础的问题都搞错实在太丢人! 综合网上的描述,我大概了解了是怎么回事,现在整理如下,如有不对之处望大神提出! 先来看一个作为程序员都熟悉的值传递

《编程导论(Java)&amp;#183;3.3.2 按值传递语义》

不要受<Java编程思想>的影响,计算机科学中的术语--按引用传递(pass-by-reference).不要搞成自说自话的个人用语. 这些术语也不是专门针对Java的,你不应该从某一本Java书上学习 不可以用于C.C++或Fortran语言的 特殊的"按引用传递". 验证按值传递很easy.在方法体中使用一个赋值语句,将形參作为左值. 按值传递时,对形參的赋值,不会影响实參.也就是说.那个赋值语句不会有不论什么副作用. 对于foo(A a),注意方法体中你要玩的是 a=

JS基础类型和对象,分别是按值传递还是按引用传递?

在分析这个问题之前,我们需了解什么是按值传递(call by value),什么是按引用传递(call by reference).在计算机科学里,这个部分叫求值策略(Evaluation Strategy).它决定变量之间.函数调用时实参和形参之间值是如何传递的. 探究JS值的传递方式 JS的基本类型,是按值传递的: var a = 1; function foo(x) { x = 2; } foo(a); console.log(a); // 仍为1, 未受x = 2赋值所影响 再来看对象:

Java:按值传递还是按引用传递详细解说

前天在做系统的时候被Java中参数传递问题卡了一下,回头查阅了相关的资料,对参数传递问题有了新的了解和掌握,但是有个问题感觉还是很模糊,就是Java中到底是否只存在值传递,因为在查阅资料时,经常看到有人说Java只有值传递,但有人说既有值传递,也有引用传递,对于两个观点个人觉得应该是站的角度不同而得出两个不同的说法,其实两个说法其中的原理是一样的,只要咱们懂得其中的原理,那么至于叫什么也就无所谓了,下面是我在网上看到的一个帖子,解释的感觉挺全面,就转过来,以供以后学习参考: 1:按值传递是什么

PHP 数组的拷贝是按值传递 or 按引用传递

在记忆中 PHP 简单变量的拷贝是按值传递,数组和对象的拷贝是按引用传递,即通过引用来实现. 简单变量和对象好理解: <?php // 简单变量的拷贝 $a = 'human'; $b = $a; $b = 'cat'; var_dump($a); // string 'human' (length=5) // 对象的拷贝 class A{} $a = new A(); $b = $a; $b->name = 'jack'; var_dump($a); // object(A)[1] publ

按值传递对象和按址传递对象

#include<iostream> using namespace std; class A { public: A(){cout<<"执行构造函数创建一个对象\n";} A(A&){cout<<"执行复制构造函数创建该对象的副本\n";} ~A(){cout<<"执行析构函数删除该对象\n";} }; A func(A one) { return one; } int main() {