基本类型值有:undefined,NUll,Boolean,Number和String
,这些类型分别在内存中占有固定的大小空间,他们的值保存在栈空间,我们通过按值来访问的。
(1)值类型:数值、布尔值、null、undefined。
(2)引用类型:对象、数组、函数。
如果赋值的是引用类型的值,则必须在堆内存中为这个值分配空间。由于这种值的大小不固定(对象有很多属性和方法),因此不能把他们保存到栈内存中。但内存地址大小是固定的,因此可以将内存地址保存在栈内存中。
简而言之,堆内存存放引用值,栈内存存放固定类型值。“引用”是一个指向对象实际位置的指针。
在这里需注意的是,引用指向的是具体的对象,而不是另一个引用。
这里的对象可以是字符串对象,数字对象,数组对象等
复制变量值
再看下面这个例子:
由以上可以得出:在变量复制方面,基本类型和引用类型也有所不同,基本类型复制的是值本身,而引用类型复制的是地址。
传递参数
ECMAScript中,所有函数的参数都是按值传递的,
js
没有按引用传递的,如果存在引用传递的话,那么函数内的变量将是全局变量,在外部也可以访问。但这明显是不可能的。
执行环境及作用域
执行环境是javascript
中最为重要的概念之一,执行环境定义了变量或函数有权访问其他数据。
全局执行环境是最外围的执行环境,在web浏览器中,全局执行环境是window
对象,因此,所有的全局变量的函数都是作为window
的属性和方法创建的。
当执行环境内的代码执行完毕后,该环境被销毁,保存其中的变量和函数也随之销毁,如果是全局环境,需所有程序执行完毕或网页完毕后才会销毁。
去掉var的局部变量
通过传参,也是局部变量
函数体内还包含函数,只有这个函数才可以访问内一层的函数
可以通过如下方法进行访问:
再一个作用域例子:
当代码在一个环境中执行的时候,就会形成一种叫做作用域链
的东西,它的用途是保证对执行环境中有访问权限的变量和函数进行有序访问(指按照规则层次来访问),作用域链的前端,就是执行环境的变量对象。
作用域
变量没有在函数内声明或者声明的时候没有带var
就是全局变量,拥有全局作用域,window
对象的所有属性拥有全局作用域;在代码任何地方都可以访问,函数内部声明并且以var
修饰的变量就是局部变量,只能在函数体内使用,函数的参数虽然没有使用var
但仍然是局部变量。
没有块级作用域
- // if语句:
for循环语句也是如此。
变量的查询
在变量的查询中,访问局部变量要比全局变量来得快,因此不需要向上搜索作用域链。
如下例子:
每个环境都可以向上搜索作用域链,以查询变量和函数名;但任何环境都不能通过向下搜索作用域链而进入另一个执行环境。在这里,如果去掉var name = "trigkit4"
,那么将弹出“Jack”
内存问题
javascript
具有自动垃圾回收机制,一旦数据不再使用,可以将其设为”null”来释放引用
循环引用
一个很简单的例子:一个DOM对象被一个Javascript
对象引用,与此同时又引用同一个或其它的Javascript
对象,这个DOM
对象可能会引发内存泄露。这个DOM
对象的引用将不会在脚本停止的时候被垃圾回收器回收。要想破坏循环引用,引用DOM
元素的对象或DOM
对象的引用需要被赋值为null
。
闭包
在闭包中引入闭包外部的变量时,当闭包结束时此对象无法被垃圾回收(GC)。
var a = function() {
var largeStr = new Array(1000000).join('x');
return function() {
return largeStr;
}
}();
DOM泄露
当原有的COM被移除时,子结点引用没有被移除则无法回收。
var select = document.querySelector;
var treeRef = select('#tree');
//在COM树中leafRef是treeFre的一个子结点
var leafRef = select('#leaf');
var body = select('body');
body.removeChild(treeRef);
//#tree不能被回收入,因为treeRef还在
//解决方法:
treeRef = null;
//tree还不能被回收,因为叶子结果leafRef还在
leafRef = null;
//现在#tree可以被释放了。
Timers计(定)时器泄露
定时器也是常见产生内存泄露的地方:
- for (var i = 0; i < 90000; i++) {
- var buggyObject = {
- callAgain: function() {
- var ref = this;
- var val = setTimeout(function() {
- ref.callAgain();
- }, 90000);
- }
- }
- buggyObject.callAgain();
- //虽然你想回收但是timer还在
- buggyObject = null;
- }
调试内存
Chrome
自带的内存调试工具可以很方便地查看内存使用情况和内存泄露,在 Timeline -> Memory 点击record即可。
分享标题:JavaScript变量、作用域及内存详解
URL标题:http://www.shufengxianlan.com/qtweb/news8/427458.html
网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联