`
sntetwt
  • 浏览: 20126 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
最近访客 更多访客>>
社区版块
存档分类
最新评论

JavaScript定义类或函数的几种方式小结

阅读更多

js中不论是定义类或者函数,很多朋友想将代码写的更专业,更方便扩展等,那么就可以参考这篇文章了,最好是总结,建议大家收藏下。

提起面向对象我们就能想到类,对象,封装,继承,多态。在《javaScript高级程序设计》(人民邮电出版社,曹力、张欣译。英文名字是:Professional JavaScript for Web Developers)这本书中描述的还算比较详细。我们看看JavaScript中定义类的各种方法。
1.工厂方式
javaScript中创建自己的类和对象,我们应该是必须掌握的,我们都知道javaScript中对象的属性可以在对象创建后动态定义,比如下面的代码:

<script type="text/javascript"> 
//定义 
var oCar = new Object(); 
oCar.color = "red"; 
oCar.doors = 4; 
oCar.showColor = function() { 
alert(this.color); 
} 
//调用 
oCar.showColor(); 
</script> 

 我们很容易使用oCar对象,但是我们创就是想创建多个Car实例。我们可以使用一个函数来封装上面的代码来实现:

//定义 
function createCar() { 
var oCar = new Object(); 
oCar.color = "red"; 
oCar.doors = 4; 
oCar.showColor = function() { 
alert(this.color); 
} 
return oCar; 
} 
//调用 
var ocar1 = createCar(); 
var ocar2 = createCar(); 
ocar1.color = "black"; 
ocar1.showColor(); 
ocar2.showColor(); 
</script> 

 

顺便说一下,javaScript对象默认成员属性都是public 的。这种方式我们称为工厂方式,我们创造了能创建并返回特定类型的对象的工厂。
这样做有点意思了,但是在面向对象中我们经常使用创建对象的方法是:
Car car=new Car();
使用new 关键字已经深入人心,因此我们使用上面的方法去定义总感觉别扭,并且每次调用时都去创建新的属性以及函数,功能上也不实际。下来我们看看构造函数的形式定义类。
2.构造函数
这种方式看起来有点象工厂函数。具体表现如下:

<script type="text/javascript"> 
//定义 
function Car(color, doors) { 
this.color = color; 
this.doors = doors; 
this.showColor = function() { 
alert(this.color); 
}; 
} 
//调用 
var car1 = new Car("red", 4); 
var car2 = new Car("blue", 4); 
car1.showColor(); 
car2.showColor(); 
</script> 

 

看起来效果很明显,有差别了吧。感觉有点意思了。在构造函数内部创造对象使用this 关键字,使用new 运算符创建对象感觉非常亲切。但是也有点问题:每次new 对象时都会创建所有的属性,包括函数的创建,也就是说多个对象完全独立,我们定义类的目的就是为了共享方法以及数据,但是car1对象与car2对象都是各自独立的属性与函数,最起码我们应该共享方法。这就是原形方式的优势所在。
3.原型方式
利用对象的prototype属性,可把它看出创建新对象所依赖的原型。方法如下:

<script type="text/javascript"> 
//定义 
function Car() { 
}; 
Car.prototype.color = "red"; 
Car.prototype.doors = 4; 
Car.prototype.drivers = new Array("Tom", "Jerry"); 
Car.prototype.showColor = function() { 
alert(this.color); 
} 
//调用: 
var car1 = new Car(); 
var car2 = new Car(); 
car1.showColor(); 
car2.showColor(); 
alert(car1.drivers); 
car1.drivers.push("stephen"); 
alert(car1.drivers); //结果:Tom,Jerry,stephen 
alert(car2.drivers); //结果:Tom,Jerry,stephen 
//可以用json方式简化prototype的定义: 
Car.prototype = 
{ 
color: "red", 
doors: 4, 
drivers: ["Tom", "Jerry",'safdad'], 
showColor: function() { 
alert(this.color); 
} 
}
</script> 

 首先这段代码的构造函数,其中没有任何代码,接下来通过对象的prototype属性添加属性定义Car对象的属性。这种方法很好,但是问题是Car的对象指向的是Array指针,Car的两个对象都指向同一个Array数组,其中一个对象car1改变属性对象的引用(数组Array)时,另一个对象car2也同时改变,这是不允许的。
同时该问题也表现在原型不能带任何初始化参数,导致构造函数无法正常初始化。这需要另一种方式来解决:那就是混合的构造函数/原型模式。
4. 混合的构造函数/原型模式
联合使用构造函数和原型方式,定义类就非常方便。

<script type="text/javascript"> 
//定义 
function Car(color,doors) 
{ 
this.color=color; 
this.doors=doors; 
this.drivers=new Array("Tom","Jerry"); 
} 
Car.prototype.showColor=function(){ 
alert(this.color); 
} 
//调用: 
var car1=new Car('red',4); 
var car2=new Car('blue',4); 
car1.showColor(); 
car2.showColor(); 
alert(car1.drivers); 
car1.drivers.push("stephen"); 
alert(car1.drivers); //结果:Tom,Jerry,stephen 
alert(car2.drivers); //结果:Tom,Jerry 
alert(car1 instanceof Car); 
</script> 

 

该方法是把属性放在内部定义,把方法放在外边利用prototype进行定义。解决了第三种方法的问题。
这种方法其实应该来说非常友好了,但是比起java的语法来,应该有一些不和谐,感觉比较凌乱,对C++来说,我们就没有那么麻烦的感觉了,可是开发C++的研发人员一般情况下很少涉及javaScript,而对J2EE的研发人员来说,这种方式总有一些别扭。总感觉不是友好的封装,其实只不过是视觉上封装效果不是很好而已,要想达到视觉封装效果而又能达到这种方法的效果的也可以以,个人认为其实比较麻烦。那就是动态原型法。
5.动态原型
对于习惯使用其他语言的开发者来说,使用混合的构造函数/原型方式感觉不那么和谐。毕竟,定义类时,大多数面向对象语言都对属性和方法进行了视觉上的封装。考虑下面的C#类:

class Car //class 
{ 
public string color = "red"; 
public int doors = 4; 
public int mpg = 23; 
public Car(string color, int doors, int mpg) //constructor 
{ 
this.color = color; 
this.doors = doors; 
this.mpg = mpg; 
} 
public void showColor() //method 
{ 
Console.WriteLine(this.color); 
} 
} 

 

C#很好的打包了Car类的所有属性和方法,因此看见这段代码就知道它要实现什么功能,它定义了一个对象的信息。批评混合的构造函数/原型方式的人认为,在构造函数内存找属性,在其外部找方法的做法不合逻辑。因此,他们设计了动态原型方法,以提供更友好的编码风格。
动态原型方法的基本想法与混合的构造函数/原型方式相同,即在构造函数内定义非函数属性,而函数属性则利用原型属性定义。唯一的区别是赋予对象方法的位置。下面是用动态原型方法重写的Car类:

<script type="text/javascript"> 
//定义 
function Car() { 
this.color = "red"; 
this.doors = 4; 
this.drivers = new Array("Tom", "Jerry"); 
if (typeof Car._initialized == "undefined") { 
Car.prototype.showColor = function() { 
alert(this.color); 
} 
//............ 
} 
//最后定义 
Car._initialized = true; 
} 
</script> 

 

直到检查typeof Car._initialized是否等于"undefined"之前,这个构造函数都未发生变化。这行代码是动态原型方法中最重要的部分。如果这个值未定义,构造函数将用原型方式继续定义对象的方法,然后把Car._initialized设置为true。如果这个值定义了(它的值为true时,typeof的值为Boolean),那么就不再创建该方法。简而言之,该方法使用标志(_initialized)来判断是否已给原型赋予了任何方法。该方法只创建并赋值一次,为取悦传统的OOP开发者,这段代码看起来更像其他语言中的类定义了。
6 混合工厂方式

这种方式通常是在不能应用前一种方式时的变通方法。它的目的是创建假构造函数,只返回另一种对象的新实例。这段代码看来与工厂函数非常相似:

function Car() { 
var oTempCar = new Object(); 
oTempCar.color="red"; 
oTempCar.doors=4; 
oTempCar.mpg=23; 
oTempCar.showColor = function() { 
alert(this.color); 
} 
return oTempCar; 
} 

 

与经典方式不同,这种方式使用new运算符,使它看起来像真正的构造函数:
var oCar = new Car();

由于在Car()构造函数内部调用了new运算符,所以将忽略第二个new运算符(位于构造函数之外)。在构造函数内部创建的对象被传递回变量var。这种方式在对象方法的内部管理方面与经典方式有着相同的问题。强烈建议:除非万不得已(请参阅第15章),还是避免使用这种方式。
总结:(采用哪种方式)
目前使用最广泛的是混合的构造函数/原型方式。此外,动态原型方法也很流行,在功能上与构造函数/原型方式等价。可以采用这两种方式中的任何一种。不过不要单独使用经典的构造函数或原型方式,因为这样会给代码引入问题。

//ps 
//static class (1:function) 
var CarCollection = new function() { 
var _carCollection = new Array(); //global,private 
this.Add = function(objCar) { 
alert('Add'); 
} 
this.Get = function(carid) { 
alert('Get'); 
} 
} 
//static class (2:json) 

var Car = { 
color: 'red', 
doors: 4, 
showColor: function() { alert(this.color); } 
} 
Car.showColor(); 

 

分享到:
评论

相关推荐

    JS定义函数的几种常用方法小结

    本文实例讲述了JS定义函数的几种常用方法。分享给大家供大家参考,具体如下: 在 JavaScript 语言里,函数是一种对象,所以可以说函数是 JavaScript 里的一等公民(first-class citizens)。 之前我们这样定义过一个...

    javascript函数定义的几种区别小结

    本篇文章主要是对javascript函数定义的几种区别进行了详细的总结介绍,需要的朋友可以过来参考下,希望对大家有所帮助

    php网络开发完全手册

    1.5 几种综合网络服务器系统的安装 14 1.5.1 XAMPP 14 1.5.2 WAMP 16 1.5.3 Appserv 17 1.5.4 EasyPHP 18 1.5.5 VertrigoServ 19 1.6 几种开发工具的介绍 20 1.6.1 Vi及Vim 20 1.6.2 Eclipse+PHPEclipse插件 21 ...

    JavaScript中的this基本问题实例小结

    本文实例讲述了JavaScript中的this基本问题.分享给大家供大家参考,具体如下: ...关于 this 的取值,大体上可以分为以下几种情况: 情况一:全局 & 调用普通函数 在全局环境中,this 永远指向 window。 c

    JS常见创建类的方法小结【工厂方式,构造器方式,原型方式,联合方式等】

    本文实例讲述了JS常见创建类的方法。分享给大家供大家参考,具体如下: Javascript是一种基于对象的...接下来我们介绍一下在JS中创建类的几种方式: 1.工厂方式: //通过工厂方式创建对象,先定义一个工厂方法 functi

    jQuery权威指南366页完整版pdf和源码打包

    3.10.3 功能实现 3.10.4 代码分析 3.11 本章小结 第4章 jquery中的事件与应用 4.1 事件机制 4.2 页面载入事件 4.2.1 ready()方法的工作原理 4.2.2 ready()方法的几种相同写法 4.3 绑定事件 ...

    网页与Web程序设计 课件 ppt 机械工业出版社 part1

    15.1.1 创建逐帧动画的几种方法 284 15.1.2 制作过程 284 15.2 形状补间动画 287 15.2.1 创建形状补间动画的方法 287 15.2.2 制作过程 287 15.3 动作补间动画 288 15.3.1 创建动作补间动画的方法 288 15.3.2...

    jQuery权威指南-源代码

    4.2.2 ready()方法的几种相同写法/86 4.3 绑定事件/86 4.4 切换事件/90 4.4.1 hover()方法/90 4.4.2 toggle()方法/93 4.5 移除事件/94 4.6 其他事件/96 4.6.1 方法one() /97 4.6.2 方法trigger () /98 4.7...

    亮剑.NET深入体验与实战精要2

    本章小结 77 第2章 细节决定成败 79 2.1 Equals()和运算符==的区别 80 2.2 const和readonly的区别 82 2.3 private、protected、public和internal的区别 86 2.4 sealed、new、virtual、abstract与override 87 2.5 ...

    亮剑.NET深入体验与实战精要3

    本章小结 77 第2章 细节决定成败 79 2.1 Equals()和运算符==的区别 80 2.2 const和readonly的区别 82 2.3 private、protected、public和internal的区别 86 2.4 sealed、new、virtual、abstract与override 87 2.5 ...

    asp.net知识库

    C#静态成员和方法的学习小结 C#中结构与类的区别 C#中 const 和 readonly 的区别 利用自定义属性,定义枚举值的详细文本 Web标准和ASP.NET - 第一部分 XHTML介绍 在ASP.NET页面中推荐使用覆写(Override)而不是事件...

    工程硕士学位论文 基于Android+HTML5的移动Web项目高效开发探究

    2.4 本章小结 15 第三章 跨域交互缓存处理设计 17 3.1 跨域交互缓存处理需求 17 3.1.1 缓存技术WebStorage 17 3.1.2 跨域交互缓存处理需求 17 3.1.3 页面回退管理需求 17 3.2 缓存处理机制 18 3.2.1 跨域缓存处理 18...

    jquery插件使用方法大全

    jQuery.sub() jQuery 1.5提供了一种创建和修改jQuery副本的方式。可以用来添加不向外部公开的方法,或者对jQuery的某些方法进行重新定义以提供新功能,或者提供更好的封装、避免名称空间冲突。当然,也可以用来开发...

Global site tag (gtag.js) - Google Analytics