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 相关文章推荐
Javascript实现的分页函数
Feb 07 Javascript
JS实现程序暂停与继续功能代码解读
Oct 10 Javascript
JQuery插入DOM节点的方法
Jun 11 Javascript
去除字符串左右两边的空格(实现代码)
May 12 Javascript
微信开发 消息推送实现代码
Oct 21 Javascript
jquery 手势密码插件
Mar 17 Javascript
vue 使用ref 让父组件调用子组件的方法
Feb 08 Javascript
浅谈React组件之性能优化
Mar 02 Javascript
postman自定义函数实现 时间函数的思路详解
Apr 17 Javascript
node使用mysql获取数据库数据中文乱码问题的解决
Dec 02 Javascript
微信小程序实现列表左右滑动
Nov 19 Javascript
vue中axios封装使用的完整教程
Mar 03 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
php桌面中心(一) 创建数据库
2007/03/11 PHP
支持中文字母数字、自定义字体php验证码代码
2012/02/27 PHP
php内核解析:PHP中的哈希表
2014/01/30 PHP
php采集中国代理服务器网的方法
2015/06/16 PHP
input+select(multiple) 实现下拉框输入值
2009/05/21 Javascript
jQuery基础框架浅入剖析
2012/12/27 Javascript
JS取request值以及自动执行使用示例
2014/02/24 Javascript
JS循环遍历JSON数据的方法
2014/07/08 Javascript
jQuery mobile在页面加载时添加加载中效果 document.ready 和window.onload执行顺序比较
2016/07/14 Javascript
AngularJS中的Promise详细介绍及实例代码
2016/12/13 Javascript
angular 动态组件类型详解(四种组件类型)
2017/02/22 Javascript
原生js实现放大镜特效
2017/03/08 Javascript
Vue.js数据绑定之data属性
2017/07/07 Javascript
微信小程序实现人脸检测功能
2018/05/25 Javascript
在vue中读取本地Json文件的方法
2018/09/06 Javascript
一百行JS代码实现一个校验工具
2019/04/30 Javascript
浅谈Node 异步IO和事件循环
2019/05/05 Javascript
小程序rich-text组件如何改变内部img图片样式的方法
2019/05/22 Javascript
详解vue-property-decorator使用手册
2019/07/29 Javascript
vue 自定义组件的写法与用法详解
2020/03/04 Javascript
利用Python的Twisted框架实现webshell密码扫描器的教程
2015/04/16 Python
python实现在sqlite动态创建表的方法
2015/05/08 Python
Python设计模式之工厂模式简单示例
2018/01/09 Python
python读取dicom图像示例(SimpleITK和dicom包实现)
2020/01/16 Python
配置python的编程环境之Anaconda + VSCode的教程
2020/03/29 Python
在django中实现choices字段获取对应字段值
2020/07/12 Python
Django视图、传参和forms验证操作
2020/07/15 Python
关于h5中的fetch方法解读(小结)
2017/11/15 HTML / CSS
HTML5实现移动端点击翻牌功能
2020/10/23 HTML / CSS
台湾最大网路书店:博客来
2018/03/18 全球购物
美国男士和女士奢侈品折扣手表购物网站:Certified Watch Store
2018/06/13 全球购物
奥地利体育网上商店:Gigasport
2019/10/09 全球购物
《夕阳真美》教学反思
2014/04/27 职场文书
体育运动会广播稿
2014/10/05 职场文书
2015驻村干部工作总结
2015/04/07 职场文书
SpringBoot SpringEL表达式的使用
2021/07/25 Java/Android