JS类的封装及实现代码


Posted in Javascript onDecember 02, 2009

1. 定义js类
js并不是一种面向对向的语言, 没有提供对类的支持, 因此我们不能像在传统的语言里那样 用class来定义类, 但我们可以利用js的闭包封装机制来实现js类, 我们来封装一个简的Shape类.

function ShapeBase() { 
this.show = function() 
{ 
alert("ShapeBase show"); 
}; 
this.init = function(){ 
alert("ShapeBase init"); 
}; 
}

这个类里定义了两个方法:show和init, 需要注意的是这里用到了this来声明, 而不是var, 因为用var是用来定义私有方法的.
另外, 我们还可以用prototype属性来定义Shape的方法.
ShapeBase.prototype.show=function() 
{ 
alert("ShapeBase show"); 
} 
ShapeBase.prototype.init=function() 
{ 
alert("ShapeBase init"); 
}

上面这种写法看起来不太直观,我们可以将所有的方法写在一起.
ShapeBase.prototype={ 
show:function() 
{ 
alert("ShapeBase show"); 
}, 
init:function() { 
alert("ShapeBase init"); 
} 
};

现在, 类是写好了, 让我们写个js来测试下, 看看结果是不是跟我们想象的一样呢?
function test(src){ 
var s=new ShapeBase(); 
s.init(); 
s.show(); 
}

看到了吧, 其调用方式和C#一模一样, 而结果也如我们所料.
到目前为止, 我们学会了如何创建js的类了, 但还只是实例方法,要是实现跟C#中的静态方法要怎么做呢?
其实, 实现js的静态方法很简单, 看下面如何实现:
//静态方法 
ShapeBase.StaticDraw = function() 
{ 
alert("method draw is static"); 
}

2. 实现JS类抽象和继承
同样, js中也不支持类继承机制,但我们可以通过将父类prototype中的成员方法复制到子类的prototype中来实现.
和类的继承一样,JavaScript也没有任何机制用于支持抽象类.但利用JavaScript语言本身的性质.可以实现自己的抽象类.
首先来看看js中的虚方法, 在传统语言中虚方法是要先定义的, 而包含虚方法的类就是抽象类,不能被实例化,而在JavaScript中,虚方法就可以看作该类中没有定义的方法,但已经通过this指针使用了.
和传统面向对象不同的是,这里虚方法不需经过声明,而直接使用了, 并且类也可以被实例化.
先定义object的extend方法, 一个为静态方法,一个为实例方法, 这两个方法用于实现继承的prototype复制
Object.extend = function(destination, source) { 
for (property in source) { 
destination[property] = source[property]; 
} 
return destination; 
} 
Object.prototype.extend = function(object) { 
return Object.extend.apply(this, [this, object]); 
}

接下来我们实现一个继承类Rect, 这里先用一种简单的方法来实现。
function Rect() { 
} 
Rect.prototype = ShapeBase.prototype; //只这一句就行了 
//扩充新的方法 
Rect.prototype.add=function() { 
alert("Rect add"); 
}

这种方法不能用于重写,如果改变了show方法, ShapeBase的show也会指向同一函数可能是由于prototype赋值只是简单的改变指向地址.
如果上面也定义了:
Rect.prototype.show=function() {
alert("Rect show");
}
那么执行结果如下:
function test(){
var s=new ShapeBase();
s.show(); //结果:Rect show
var r=new Rect();
r.show(); //结果:Rect show
r.add();
}
我们再使用object.extend实现继承, 并实现一个oninit虚方法, 修改ShapeBase如下:
ShapeBase.prototype={ 
show:function() 
{ 
alert("ShapeBase show"); 
}, 
initialize:function () { 
this.oninit(); 
} 
};

实现Rect类继承.
Rect.prototype=(new ShapeBase).extend({ 
//添加新的方法 
add:function() { 
alert("Rect add"); 
}, 
//使用这种方法可以重写show方法 
show:function() { 
alert("Rect show"); 
}, 
//实现虚方法 
oninit:function() { 
alert("Rect oninit"); 
} 
})

现在我们的类写好了, 测试下看看:
function test(src){ 
ShapeBase.StaticDraw(); 
var s=new ShapeBase(); 
s.show(); //alert("ShapeBase show") 
var r=new Rect(); 
r.show(); //alert("Rect show") 
r.add(); 
r.initialize(); //alert("Rect oninit") 
}

另外,在网上看到一篇用专门的对象来创建类,代码如下:
// 
//对象属性复制方法,很多库都有实现,如PrototypeJS里面的extend和Ext里面的Ext.apply 
// 
function extend(des, src) { 
if (!des) 
des = {}; 
if (src) { 
for (var i in src) { 
des[i] = src[i]; 
} 
} 
return des; 
} 
var CC = {}; //全局变量 
// 
//create 用于创建类 
// 
CC.create = function(superclass, constructor){ 
var clazz = (function() { 
this.initialize.apply(this, arguments); 
}); 
//如果无参数,直接返回类. 
if(arguments.length == 0) 
return clazz; 
//如果无父类,此时constructor应该为一个纯对象,直接复制属性返回. 
if(!superclass){ 
extend(clazz.prototype, constructor); 
return clazz; 
} 
var absObj = clazz.prototype, 
sprPropty = superclass.prototype; 
if(sprPropty){ 
//用于访问父类方法 
clazz.superclass = sprPropty; 
extend(absObj, sprPropty); 
//调用属性构造函数创建属性,这个是实现关键. 
extend(absObj, constructor(sprPropty)); 
// 子类实例直接通过obj.superclass访问父类属性, 
// 如果不想造成过多引用,也可把这句注释掉,因为多数时候是没必要的. 
absObj.superclass = sprPropty; 
// 
clazz.constructor = constructor; 
} 
return clazz; 
} 
// 
//创建一个动物类 
// 
var Animal = CC.create(null, { 
//属性 
footprint : '- - - - - - =', 
//类初始化方法,必须的,当用 new 生成一个类时该方法自动被调用,参见上定义. 
initialize : function(options){ 
extend(this, options); 
alert('Animal initialize method is called.'); 
}, 
eat : function(){ 
alert('Animal eat method is called.'); 
}, 
move : function(){ 
alert('I am moving like this '+ this.footprint +' .'); 
} 
}); 
// 
//创建一个Duke类 
// 
var Duke = CC.create(Animal, function(superclass){ 
//在这可以定义一些类全局静态数据,该类每个实例都共享这些数据. 
//计算实例个类,包括派生类实例. 
var static_instance_counter = 0; 
function classUtilityFuncHere(){ } 
//返回类具体属性. 
return { 
//重写初始化方法 
//@override 
initialize : function(options) { 
alert('Initializing Duke class..'); 
//调用父类初始化,这种方法比一般其它库的要简洁点吧,可以不管父类是什么. 
superclass.initialize.call(this, options); 
//做一些子类喜欢做的事. 
alert('Duke initialize method is called.'); 
//读取或修改类静态属性 
static_instance_counter++; 
}, 
//重写move方法,增加Duke自己的移动方式. 
move : function(){ 
this.footprint = this.footprint + 'zzzzzzzz'; 
superclass.move.call(this); 
}, 
//重写eat方法,注意,里面不调用父类方法,即父类eat被覆盖了. 
eat : function(){ 
alert('Duke is eating..'); 
}, 
//新增一个say方法,显示当前已经初始化的Duke类实例数量. 
say : function(){ 
alert('the number of Duke instances is '+static_instance_counter); 
} 
}; 
}); 
var DukeChild = CC.create(Duke, function(superclass){ 
return { 
move : function(){ 
this.footprint = this.footprint + '++++++++++++='; 
superclass.move.call(this); 
}, 
say : function(){ 
alert(this.msg || ''); 
} 
}; 
}); 
(function test() { 
var animal = new Animal(); 
animal.eat(); 
animal.move(); 
var dukeA = new Duke(); 
dukeA.eat(); 
dukeA.move(); 
dukeA.say(); 
var dukeB = new Duke(); 
dukeB.eat(); 
dukeB.move(); 
dukeB.say(); 
var dukeC = new DukeChild({msg : 'I am a child of duke.'}); 
dukeC.move(); 
dukeC.say(); 
})();
Javascript 相关文章推荐
免费空间广告万能消除代码
Sep 04 Javascript
JS函数实现动态添加CSS样式表文件
Dec 15 Javascript
js模拟滚动条(横向竖向)
Feb 22 Javascript
使用jQuery清空file文件域的解决方案
Apr 12 Javascript
js中的异常处理try...catch使用介绍
Sep 21 Javascript
js实现文本框中焦点在最后位置
Mar 04 Javascript
简介AngularJS的视图功能应用
Jun 17 Javascript
jquery模拟进度条实现方法
Aug 03 Javascript
JavaScript奇技淫巧44招【实用】
Dec 11 Javascript
详解EasyUi控件中的Datagrid
Aug 23 Javascript
js实现一个简单的MVVM框架示例
Jan 15 Javascript
vue实现页面滚动到底部刷新
Aug 16 Javascript
Jquery选择器 $实现原理
Dec 02 #Javascript
js 表格隔行颜色
Dec 02 #Javascript
让FireFox支持innerText的实现代码
Dec 01 #Javascript
JavaScript 直接操作本地文件的实现代码
Dec 01 #Javascript
js 变量类型转换常用函数与代码[比较全]
Dec 01 #Javascript
Jquery 快速构建可拖曳的购物车DragDrop
Nov 30 #Javascript
JavaScript 学习笔记 Black.Caffeine 09.11.28
Nov 30 #Javascript
You might like
随时给自己贴的图片加文字的php水印
2007/03/16 PHP
PHP 冒泡排序算法的实现代码
2010/08/08 PHP
PHP中基于ts与nts版本- vc6和vc9编译版本的区别详解
2013/04/26 PHP
php+mysql实现数据库随机重排实例
2014/10/17 PHP
thinkPHP5实现数据库添加内容的方法
2017/10/25 PHP
PHP实现双链表删除与插入节点的方法示例
2017/11/11 PHP
麦鸡的TAB切换功能结合了javascript和css
2007/12/17 Javascript
学习ExtJS(一) 之基础前提
2009/10/07 Javascript
Javascript 通过json自动生成Dom的代码
2010/04/01 Javascript
js 实现打印网页中定义的部分内容的代码
2010/04/01 Javascript
Javascript之旅 对象的原型链之由来
2010/08/25 Javascript
js原型继承的两种方法对比介绍
2014/03/30 Javascript
jQuery实现设置、移除文本框默认值功能
2015/01/13 Javascript
15款jQuery分布引导插件分享
2015/02/04 Javascript
jQuery利用sort对DOM元素进行排序操作
2016/11/07 Javascript
angularjs中ng-attr的用法详解
2016/12/31 Javascript
使用JS组件实现带ToolTip验证框的实例代码
2017/08/23 Javascript
Angularjs 手写日历的实现代码(不用插件)
2017/10/18 Javascript
使用JS模拟锚点跳转的实例
2018/02/01 Javascript
微信小程序canvas实现刮刮乐效果
2018/07/09 Javascript
微信小程序用户信息encryptedData详解
2018/08/24 Javascript
jQuery中each遍历的三种方法实例分析
2018/09/07 jQuery
理解JavaScript中的Proxy 与 Reflection API
2020/09/21 Javascript
jQuery实现回到顶部效果
2020/10/19 jQuery
Python使用Mechanize模块编写爬虫的要点解析
2016/03/31 Python
浅谈Python中的作用域规则和闭包
2018/03/20 Python
python3使用Pillow、tesseract-ocr与pytesseract模块的图片识别的方法
2020/02/26 Python
python 工具 字符串转numpy浮点数组的实现
2020/03/14 Python
css3 transform属性详解
2014/09/30 HTML / CSS
金融管理专业毕业生求职信
2014/03/12 职场文书
杜甫草堂导游词
2015/02/03 职场文书
中学生自我评价2015
2015/03/03 职场文书
工作失职检讨书范文
2015/05/05 职场文书
少先队中队工作总结
2015/08/14 职场文书
导游词书写之黄山
2019/08/06 职场文书
Python基于百度API识别并提取图片中文字
2021/06/27 Python