js变量作用域及可访问性的探讨


Posted in Javascript onNovember 23, 2006

每一种语言都有变量的概念,变量是用来存储信息的一个元素。比如下面这个函数: 

 function Student(name,age,from)  
{  
 this.name = name;  
  this.age = age;  
 this.from = from;  
  this.ToString = function()  
 {  
  return "my information is name: "+this.name+",age : "+this.age+", from :" +this.from;  
  }  
}
 
   Student类有三个变量,分别为name(名字),age(年龄),from(籍贯),这三个变量构成了描述一个对象的信息。当然,这里还有一个方法用来返回Student的信息。
   但是,我们是不是定义了一个变量,它就能一直存在着,并且还有可能在任何地方都能被访问和使用直到其被销毁?仔细想想,上面的需求是比较过分的,因为某些变量在某个功能实现后就不再利用了,但如果这个变量还存在的话,就占用了系统资源了,俗语曰:“站着茅坑不拉#$%”。
   于是我们对变量的及时和按需求地销毁有一个探讨的话题了。
   好,切入正题吧,就本人所接触过的来讲,js中支持如下几种类型的变量,分别为:局部变量、类变量、私有变量、实例变量、静态变量和全局变量。接下来我们就一一探讨研究下。

局部变量:

局部变量一般指在{}范围内有效变量,也就是语句块内有效的变量,如: 

 

function foo(flag)  
{  
 var sum = 0;  
 if(flag == true)  
 {  
  var index;  
  for(index=0;index<10;index++)  
  {  
   sum +=index;  
  }  
}  
 document.write("index is :"+index+"<br>");  
 return sum;  
}  
//document.write("sum is :" +sum+"<br>");  
document.write("result is :"+foo(true)+"<br>"); 
   该代码执行后输出的结果为:“index is :undefined” 和 “result is :0”,我们可以看到希望输出的index变量的值为undefined,也就是未定义。因此我们可以发现,index变量在if语句块结束后即被销毁了。那么“sum”变量呢?这个变量在foo()函数段执行完毕后被销毁了,如果您去掉我注释的那条语句,再执行,您将会发现系统将报错。值得注意的是,如果我把上面的foo()函数改成如下: 

 

function foo(flag)  
 {  
  var sum = 0;  
  for(var index=0;index<10;index++)  
  {  
   sum +=index;  
  }  
  document.write("index is :"+index+"<br>");  
  return sum;  
} 

   您将可以看见可以输出index值("index is :10"),这个是js和其他语言的不同地方,因为index是在for循环的{}外面定义的,因此其作用范围在foo()函数使用完毕后才销毁。

  类变量:
   类变量,实际上就是类的一个属性或字段或一个方法,该变量在该类的一个实例对象被销毁后自动销毁,比如我们开始时举的Student类。这个我们不多讨论,大家可以自己试一下。

私有变量:
   私有变量,值得是某个类自己内部是用的一个属性,外部无法调用,其定义是用 var 来声明的。注意如果不用var 来声明,该变量将是全局变量(我们下面将会讨论),如: 

function Student(name,age,from)  
{   this.name = FormatIt(name);  
 this.age = age;  
this.from = from;  
 var origName = name;  
 var FormatIt = function(name)  
 {  
 return name.substr(0,5);  
 }  
 this.ToString = function()  
 {  
  return "my information is name: "+origName+",age : "+this.age+", from :" +this.from;  
 }  
} 

   这里,我们分别定义了一个origName和FormatIt()两个私有变量(按面向对象的解释,应该用类的属性来称呼)。
   我们把这种情况下的方法也成为变量,因为该情况下的变量是个function类型的变量,而function也属于Object类的继承类。在这种情形下,如果我们定义了 var zfp = new Student("3zfp",100,"ShenZhen")。但无法通过zfp.origName和zfp.FormatIt()方式来访问这两个变量的。

注意以下几点:

1、私有变量是不能用this来指示的。
2、私有方法类型的变量的调用必须是在该方法声明后。如我们将Student类改造如下:

function Student(name,age,from)  
{  
 var origName = name;  
 this.name = FormatName(name);  
 this.age = age;  
 this.from = from;  
 var FormatName = function(name)  
 {  
  return name+".china";  
 }  
 this.ToString = function()  
 {  
  return "my information is name: "+origName+",age : "+this.age+", from :" +this.from;  
 }  
}  
var zfp = new Student("3zfp",100,"ShenZhen"); 
代码执行后,将会报"找不到对象"的错误.意思是FormatName()未定义。

3、私有方法无法访问this指示的变量(公开变量),如下:

function Student(basicinfo)  
{  
 this.basicInfo = basicinfo;   var FormatInfo = function()  
 {  
  this.basicInfo.name = this.basicInfo.name+".china";  
 }  
 FormatInfo();  
}  
function BasicInfo(name,age,from)  
{  
 this.name = name;  
 this.age = age;  
 this.from = from;  
}  
var zfp = new Student(new BasicInfo("3zfp",100,"ShenZhen")); 
执行代码后,系统将会提示 “this.basicInfo为空或不是对象”的错误。
基本结论是,私有方法只能访问私有属性,私有属性在声明并赋值后可以在类的任何地方访问,

实例变量:
实例变量即某个实例对象所拥有的变量。如:

function BasicInfo(name,age,from)  
{  
 this.name = name;  
 this.age = age;  
 this.from = from;  
}  
var basicA = new BasicInfo("3zfp",100,"ShenZhen");  
basicA.generalInfo = "is 3zfp owned object";  
document.write("basicA's generalInfo is : "+ basicA.generalInfo+"<br>");  
var basicB = new BasicInfo("zfp",100,"ShenZhen");  
document.write("basicB's generalInfo is : "+ basicB.generalInfo+"<br>");  
执行该代码后,我们将可以看到如下结果:  
basicA's generalInfo is : is 3zfp owned object  
basicB's generalInfo is : undefined 
静态变量:

静态变量即为某个类所拥有的属性,通过 类名+"."+静态变量名 的方式访问该属性。如下可以做清晰的解释:

function BasicInfo(name,age,from)  
{  
 this.name = name;  
 this.age = age;  
 this.from = from;  
}  
BasicInfo.generalInfo = "is 3zfp owned object";  
var basic = new BasicInfo("zfp",100,"ShenZhen");  
document.write(basic.generalInfo+"<br>");  
document.write(BasicInfo.generalInfo+"<br>");  
BasicInfo.generalInfo = "info is changed";  
document.write(BasicInfo.generalInfo+"<br>"); 

执行以上代码,将会得到如下结果:
undefined
is 3zfp owned object
info is changed

注意以下几点:
1、以 类名+"."+静态变量名 的方式来声明一个静态变量
2、静态变量并不属于类的某个实例对象所独有的属性,为对象的共享.
3、能以实例对象名+"."+静态变量名来访问。

全局变量:
全局变量即整个系统运行期间有效访问控制的变量,通常是在一个js代码开头定义,如:

var copyright = "3zfp owned";  
var foo =function()  
{  
 window.alert(copyright);  
} 
注意以下几点:
1、如果一个变量不用var 来声明,则其被视为全局变量。如:
var copyright = "3zfp owned";
var foo =function(fooInfo)
{
 _foo = fooInfo;
document.write(copyright+"<br>");
}
new foo("foo test");
document.write(_foo+"<br>");
执行代码,将得到如下结果:
3zfp owned
foo test
但是,这个又有一个注意的地方,function是编译期对象,也就是说_foo这个全局变量要在foo对象被实例化后才能被初始化,也就是说如果将
new foo();
document.write(_foo+"<br>");
对调成
document.write(_foo+"<br>");
new foo();
系统将提示 "_foo 未定义"。 
2、如果定义了一个和全局变量同名的局部变量属性,如下:
var copyright = "3zfp owned";  
var foo =function(fooInfo)  
{  
 var copyright = fooInfo; //同名变量  
 this.showInfo = function()  
 {  
 document.write(copyright+"<br>");  
 }  
}  
new foo("foo test").showInfo();  
document.write(copyright+"<br>"); 
执行代码,将得到如下结果:
3zfp owned
foo test

原因是由于function 是在编译期间完成变量的定义,也就是foo内部的copyright的定义是在编译期间完成的,其作用域只在foo对象内有效,而与外部定义的全局变量copyright无关。

Javascript 相关文章推荐
JS window.opener返回父页面的应用
Oct 24 Javascript
基于javascript的JSON格式页面展示美化方法
Jul 02 Javascript
深入理解JavaScript系列(28):设计模式之工厂模式详解
Mar 03 Javascript
jQuery Mobile弹出窗、弹出层知识汇总
Jan 05 Javascript
JS实现页面载入时随机显示图片效果
Sep 07 Javascript
js中小数向上取整数,向下取整数,四舍五入取整数的实现(必看篇)
Feb 13 Javascript
vue router-link传参以及参数的使用实例
Nov 10 Javascript
微信小程序模板(template)使用详解
Jan 31 Javascript
JS/HTML5游戏常用算法之追踪算法实例详解
Dec 12 Javascript
vue实现下拉加载其实没那么复杂
Aug 13 Javascript
如何用JavaScript学习算法复杂度
Apr 30 Javascript
vue.js Router中嵌套路由的实用示例
Jun 27 Vue.js
bcastr2.0 通用的图片浏览器
Nov 22 #Javascript
一种JavaScript的设计模式
Nov 22 #Javascript
表单内同名元素的控制
Nov 22 #Javascript
document.all与WEB标准
May 13 #Javascript
云网广告中的代码,提示出错,大家找找
Nov 21 #Javascript
在视频前插入广告
Nov 20 #Javascript
使两个iframe的高度与内容自适应,且相等
Nov 20 #Javascript
You might like
完美解决:Apache启动问题―(OS 10022)提供了一个无效的参数
2013/06/08 PHP
JS图片无缝滚动(简单利于使用)
2013/06/17 Javascript
使用javascript实现json数据以csv格式下载
2015/01/09 Javascript
Javascript实现多彩雪花从天降散落效果的方法
2015/02/02 Javascript
JS实现Fisheye效果动感放大菜单代码
2015/10/21 Javascript
js拖拽功能实现代码解析
2016/11/28 Javascript
vue-cli的eslint相关用法
2017/09/29 Javascript
nodejs结合Socket.IO实现的即时通讯功能详解
2018/01/12 NodeJs
如何基于filter实现网站整体变灰功能
2020/04/17 Javascript
JavaScript监听一个DOM元素大小变化
2020/04/26 Javascript
ES6箭头函数和扩展实例分析
2020/05/23 Javascript
Vue 实现一个简单的鼠标拖拽滚动效果插件
2020/12/10 Vue.js
一篇文章看懂JavaScript中的回调
2021/01/05 Javascript
常见的在Python中实现单例模式的三种方法
2015/04/08 Python
python模拟登录并且保持cookie的方法详解
2017/04/04 Python
Python原始字符串与Unicode字符串操作符用法实例分析
2017/07/22 Python
Python可变参数*args和**kwargs用法实例小结
2018/04/27 Python
Python实现的redis分布式锁功能示例
2018/05/29 Python
Python读取txt某几列绘图的方法
2018/10/14 Python
使用urllib库的urlretrieve()方法下载网络文件到本地的方法
2018/12/19 Python
PyQt 实现使窗口中的元素跟随窗口大小的变化而变化
2019/06/18 Python
对pyqt5中QTabWidget的相关操作详解
2019/06/21 Python
python脚本调用iftop 统计业务应用流量的思路详解
2019/10/11 Python
tensorflow实现测试时读取任意指定的check point的网络参数
2020/01/21 Python
用 python 进行微信好友信息分析
2020/11/28 Python
html5 svg 中元素点击事件添加方法
2013/01/16 HTML / CSS
阿姆斯特丹城市卡:Amsterdam Pass
2019/12/01 全球购物
西班牙在线药店:DosFarma
2020/03/28 全球购物
资产评估专业学生的自我鉴定
2013/11/14 职场文书
宾馆仓管员岗位职责
2014/07/27 职场文书
聘用意向书
2014/07/29 职场文书
行政文员实习自我鉴定范文
2014/09/14 职场文书
获奖感言怎么写
2015/07/31 职场文书
PHP使用非对称加密算法RSA
2021/04/21 PHP
CSS完成视差滚动效果
2021/04/27 HTML / CSS
Unicode中的CJK(中日韩统一表意文字)字符小结
2021/12/06 HTML / CSS