javascript高程笔记-------第四章 变量、作用域和内存问题

   首先JavaScript中的变量分为基本类型和引用类型。

 基本类型就是保存在栈内存中的简单数据段,而引用类型指的是那些保存在堆内存中的对象。

1.参数传递

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

1.1 基本数据类型的传递(undefined ,Null,Boolean,number,String)

1--->var money = 10;

2--->var t=function (money){
money = 5;
alert(money); //5
}
3--->t(money);

4--->alert(money); //10

执行过程 ①全局环境中初始化money的值 为10  ; ②创建函数 t 的执行环境  ;③ 执行 t 函数  复制全局环境中money的变量的值    赋值给 t 函数中的money    函数体内的money被改变 为5   然后再执行④  全局中的money值依然为10

1.2 对象的传递 (传递是对象的引用地址)

1--->var person = new Object();
2--->var student = person; 
3--->student.name = "zhangsan";
4--->alert(person.name); // zhangsan

 执行过程 ①全局环境中初始化person ,person引用了栈区的一个空对象  ; ②将person引用的对象地址值复制一份   赋值给student   ;③ 将student引用的对象中添加name属性 赋值为 zhangsan   ④  由于student和person引用的是同一个对象  所以值为 zhangsan!

 示例图

 


1.1 作用域链

  在浏览器中顶层的容器为window对象, 执行某段js函数时,函数内部使用的对象,首先在函数内部寻找,未找到则向上延伸至函数体外部查找,直到找到要使用的对象;

1.2 没有块级作用域

javascript 中的 if 和 for循环  内部定义的变量  将会扩充至作用域链的上一层 变量 

示例 : for循环内部定义的b和i  在aa函数中 依然可以被访问到 

function aa(){
	for(var i=0; i<5;i++){
		var b = i;
	}
  alert(i) ;   //5
  alert(b);  //4
}

1.3 变量、查询标识符

 声明方式:  ①使用var 关键字  此时变量的作用范围默认在当前执行环境内部    ② 不使用var    那么这个变量将存在余全局windon对象当中

当某个函数 引用一个变量时 ,必须通过搜索来确定引用变量的值,搜索过程从当前的执行环境逐步向上寻找整个作用域链 ,找到了该变量的定义则使用已定义的值,如未找到则认为该变量的值为undefined , 

 

var b = 5;
function a(){
	alert(b);  //5
}

 当 执行环境内部与作用域链中 存在同名变量时候  则使用离执行环境最近的值 

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

  1.4 垃圾回收

javascript具有自动收集垃圾机制,但不同的浏览器有不同的实现方式   

① 标记清除(mark-and-sweep),当变量进入执行环境(函数体内声明一个变量) 就将这个变量标记“进入环境”,而当变量离开环境时(0函数体执行完毕)标记“离开环境” ,大部分浏览器都是使用此方式进行垃圾回收 (截至2008年,IE,Firefox,Opera,Chrome,Safari)或类似回收策略,回收时间的周期不同而已

②引用计数 (reference counting): 当声明一个变量a ,将这个变量a赋值给另外一个变量b时  变量a被引用次数+1,当又将a的值赋给其他变量c时 ,a被引用的次数再+1,当程序继续执行  变量c的值 引用其他值时  ,则变量a的引用次数-1,再然后 变量b的值也不等于变量a时 ,变量a被引用的次数再-1 

当引用次数为0时,说明不会再有任何执行环境需要变量a,因此就可以回收其占用的内存,

引用计数带来的问题---------当存在互相引用时(循环引用),将无法回收被占用的内存,导致内存泄漏

function a(){
	var element = document.getElementById("somthingElement");
	var myObject = new Object();
	myObject.element = element;
	element.someObject = myObject;
}

  此时 a() 函数执行完毕后  element 与myObject互相引用  两者的引用次数都为1 因此无法释放其占用的内存 ,

不管浏览器采用如何的垃圾回收策略,程序内部最好的解决方式仍然是手动解除对象的引用:

function a(){
	var element = document.getElementById("somthingElement");
	var myObject = new Object();
	myObject.element = element;
	element.someObject = myObject;
	//coding.....
	myObject.element = null;
	element.myObject = null;
}

  在IE9中采用了纯JavaScript对象方式实现BOM和DOM  ,避免了循环引用带来的内存泄漏!

 性能优化: 最佳方式就是在执行的代码中只保存有效数据,一旦确定数据不在使用,通过设置为null的方式进行解除引用

posted @ 2017-09-29 23:59  迷~途  阅读(134)  评论(0编辑  收藏  举报