// JS变量是松散型的(不强制类型)本质,决定了它只是在特定时间用于保存特定值的一个名字而已;
// 由于不存在定义某个变量必须要保存何种数据类型值的规则,变量的值及其数据类型可以在脚本的生命周期内改变;
1 // JS变量包含两种不同的数据类型的值:基本类型值和引用类型值;2 3 // 1.基本类型值:保存在栈内存中的简单数据段;即这种值完全保存在内存中的一个位置;4 // 基本类型值包含:Undefined|Null|Boolean|Number|String;5 // 这些类型在内存中占有固定大小的空间;它们的值保存在栈空间,我们按值来访问;6 7 // 2.引用类型值:保存在堆内存中的对象(可能由多个值构成),即变量中保存的实际上只是一个指针,这个指针指向内存中的另一个位置,该位置保存对象;8 // 引用类型的值的大小不固定,因此不能保存在栈内存,必须保存在堆内存中;但可以将引用类型的值的内存地址保存在栈内存中;9 // 当查询引用类型的变量时,先从栈内存中读取内存地址,然后通过地址找到堆内存中的值;=>按引用访问;
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;
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输出的值都会被修改掉;
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函数的参数都将是局部变量;也就是说,没有按引用传递;
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;
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:当代码在一个环境中执行时,就会形成一种叫做作用域链的东西;它的用途是保证对执行环境中有访问权限的变量和函数进行有序访问;作用域链的前端,就是执行环境的变量对象;
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 }
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个变量就开始运
新闻热点
疑难解答