javascript设计模式 封装和信息隐藏(上)


Posted in Javascript onJuly 24, 2012

本文分上下两部分,上部讲基本模式(basic patterns):完全暴露法,下划线标记法和使用闭包;下部讲高级模式(Advanced Patterns),如何实现静态方法和属性,常量还有其他一些知识点。

封装是面向对象语言很基本也是很有用的特性,虽然javascript也可以称的上是面向对象语言,但他对封装的支持并不是很好,不像其他语言,只要使用private、protected就可以实现。但这并不是说就没有办法了,下面我就介绍下如何在javascript中实现封装。
一、基本模式(basic patterns),主要包括三种方式:完全暴露法,下划线标记法和使用闭包。(闭包是个很重要,也是很难的概念,有兴趣的朋友可以去网上找资料,我博客里也转载了别人的文章)。

这里我们以book类作为例子,需要创建和初始化book类。

// Book(isbn, title, author) 
var theHobbit = new Book('0-395-07122-4', 'The Hobbit', 'J. R. R. Tolkien'); 
theHobbit.display(); // Outputs the data by creating and populating an HTML element.

1.完全暴露法:
创建book类可以用最传统的构造函数方式,
var Book = function(isbn, title, author) { 
if(!this.checkIsbn(isbn)) throw new Error('Book: Invalid ISBN.'); 

this.isbn = isbn; 

//代码中 || 的作用是 如果title无值,则会把'No title specified'赋给 this.title。这种方式很好用,大家可以在自己的代码中使用。 

this.title = title || 'No title specified'; 

this.author = author || 'No author specified'; 
} 
Book.prototype = { 

//验证isbn函数 

checkIsbn: function(isbn) { 


... 

}, 

//获取isbn 

getIsbn: function() { 


return this.isbn; 

}, 

//设置isbn 

setIsbn: function(isbn) { 


if(!this.checkIsbn(isbn)) throw new Error('Book: Invalid ISBN.'); 



this.isbn = isbn; 

}, 

//获取title 

getTitle: function() { 


return this.title; 

}, 

//设置title 

setTitle: function(title) { 


this.title = title || 'No title specified'; 

}, 

//获取作者 

getAuthor: function() { 


return this.author; 

}, 

//设置作者 

setAuthor: function(author) { 


this.author = author || 'No author specified'; 

}, 

//显示函数 

display: function() { 


... 

} 
};

 代码有点多,我在这里简单讲解下。javascript中创建类和c#,java有点不同,c#,java会把所有方法和属性包在一个类文件里面,比如说
public class book() 
{ 
private string isbn; 
public string ISBN 
{ 
set 
{ 


this.isbn=value; 


 } 


 get 


 { 



return this.isbn; 


 } 

} 
... 
private bool CheckIsbn(string isdn) 
{ 
...... 
} 
...... 
public void Display() 
{ 
...... 
} 
}

javascript也可以用这种方式,但是推荐使用我上面使用的把属性定义到类定义函数(或者叫构造函数),方法定义到prototype对象中,这种做法性能要好些,至于原因大家可以去google。

上面的js代码想实现的功能是,定义一个book类,类里面包括三个私有变量(或者叫属性)isbn,title,author,一个私有方法checkIsbn,几个公有方法getIsdn,setIsdn,...display。想法是好的,但是现实是残酷的,其实那些私有属性或者方法根本一点都不私有。比如说,theHobbit.isbn = '978-0261103283';你可以用这种方式为isbn赋值,不会报错而且绝对成功。原因就是javascript没有private方式去实现对特定对象的私有化。此外这种实现方式在使用时也会造成困惑,到底类的创建者想暴露哪些属性和方法呢?下面介绍第一种改进办法,下划线标记法。

2.下划线标记法:

var Book = function(isbn, title, author) { 
// Constructor code. 

this.setIsbn(isbn); 

this.setTitle(title); 

this.setAuthor(author); 
} 
Book.prototype = { 
 

//验证isbn函数 
 

_checkIsbn: function(isbn) { 


... 

}, 
 

//获取isbn 
 

getIsbn: function() { 

 


return this._isbn; 
 

}, 
 

//设置isbn 
 

setIsbn: function(isbn) { 

 


if(!this._checkIsbn(isbn)) throw new Error('Book: Invalid ISBN.'); 


 


this._isbn = isbn; 
 

}, 
 

...
 

//显示函数 
 

display: function() { 

 

... 
 

} 
};

其实就是在所有想实现私有的属性或者方法前面加了下划线_,没别的操作。这种方法并没有实现真正的私有化,theHobbit._isbn = '978-0261103283';这样操作照样成功,这种方式最大的意义在于告诉类的使用者,作者本意上想暴露哪些对象,不想暴露哪些。但是使用者是否按照作者的想法去做,作者是控制不了的。

那有没有办法实现真正的私有化呢,答案是有的,就是利用闭包。

3.使用闭包:

javascript之所以能实现真正的封装,是和他特有的函数作用域,函数支持内部函数,还有闭包分不开的。大家可以去网上搜集相关知识加深理解。

下面首先说的就是函数作用域,在javascript中如果在一个函数内部定义了一个变量,那么函数外部是没有办法访问的。其实在javascript中实现私有属性或者方法就是利用了它这一特殊属性。例子:

function foo() { 
var a = 10; 

function bar() { 


a *= 2; 

} 

bar(); 

return a; 
}

在上面的例子中函数foo在内部定义了变量a和方法bar,在foo外部是无法访问到a和bar的,但是因为a和bar都定义在foo内部,但bar是可以访问到a的。那么有没有办法能在foo外部访问到bar呢,答案是有的,就是使用闭包。
function foo() { 
var a = 10; 

function bar() { 


a *= 2; 


return a; 

} 

return bar; 
} 
var baz = foo(); // baz is now a reference to function bar. 
baz(); // returns 20. 
baz(); // returns 40. 
baz(); // returns 80. 
var blat = foo(); // blat is another reference to bar. 
blat(); // returns 20, because a new copy of a is being used.

这就是在前面提到的javascript函数支持内部函数。内部函数bar可以访问私有变量a,函数foo又把内部函数bar抛出给baz,baz就可以访问到内部变量a了,这就实现了闭包。大家一看也就明白了,这样其实就实现了私有变量和方法。回到我们前面的book例子,实现如下:
var Book = function(newIsbn, newTitle, newAuthor) { 
// implements Publication 

// Private attributes. 

var isbn, title, author; 

// Private method. 

function checkIsbn(isbn) { 


... 

} 

// Privileged methods. 

this.getIsbn = function() { 


return isbn; 

}; 

this.setIsbn = function(newIsbn) { 


if(!checkIsbn(newIsbn)) throw new Error('Book: Invalid ISBN.'); 


isbn = newIsbn; 

}; 

this.getTitle = function() { 


return title; 

}; 

this.setTitle = function(newTitle) { 


title = newTitle || 'No title specified'; 

}; 

this.getAuthor = function() { 


return author; 

}; 

this.setAuthor = function(newAuthor) { 


author = newAuthor || 'No author specified'; 

}; 

// Constructor code. 

this.setIsbn(newIsbn); 

this.setTitle(newTitle); 

this.setAuthor(newAuthor); 
}; 
// Public, non-privileged methods. 
Book.prototype = { 

display: function() { 


... 

} 
};

上述代码就实现了 isbn, title, author和checkIsbn的私有化,外部是决定不能直接访问到的。如需访问 isbn, title, author只能通过对象级的方法getTitle,setTitle...。比如要给isbn赋值,只能用theHobbit.setIsbn = '978-0261103283';,如果你还用theHobbit._isbn = '978-0261103283';,对不起要报错了。

 好了,今天的内容就讲到这里了,希望对大家有帮助。
作者:下一站永远

Javascript 相关文章推荐
JavaScript 编程引入命名空间的方法与代码
Aug 13 Javascript
js parseInt("08")未指定进位制问题
Jun 19 Javascript
JavaScript中的View-Model使用介绍
Aug 11 Javascript
javascript日期转换 时间戳转日期格式
Nov 05 Javascript
关于jQuery中的each方法(jQuery到底干了什么)
Mar 05 Javascript
Chrome下ifame父窗口调用子窗口的问题示例探讨
Mar 17 Javascript
用js模拟struts2的多action调用示例
May 19 Javascript
举例讲解JavaScript中将数组元素转换为字符串的方法
Oct 25 Javascript
js实现table添加行tr、删除行tr、清空行tr的简单实例
Oct 15 Javascript
简单实现js无缝滚动效果
Feb 05 Javascript
node.js学习之事件模块Events的使用示例
Sep 28 Javascript
webpack打包js的方法
Mar 12 Javascript
js+xml生成级联下拉框代码
Jul 24 #Javascript
40个新鲜出炉的jQuery 插件和免费教程[上]
Jul 24 #Javascript
基于jquery的跟随屏幕滚动代码
Jul 24 #Javascript
基于JQuery的类似新浪微博展示信息效果的代码
Jul 23 #Javascript
基于jquery自定义图片热区效果
Jul 21 #Javascript
Js四则运算函数代码
Jul 21 #Javascript
Javascript对象中关于setTimeout和setInterval的this介绍
Jul 21 #Javascript
You might like
星际争霸中的热键
2020/03/04 星际争霸
神族 Protoss 历史背景
2020/03/14 星际争霸
Terran历史背景
2020/03/14 星际争霸
小偷PHP+Html+缓存
2006/11/25 PHP
php检测数组长度函数sizeof与count用法
2014/11/17 PHP
PHP输出多个元素的排列或组合的方法
2017/03/14 PHP
List Information About the Binary Files Used by an Application
2007/06/11 Javascript
javascript中获取下个月一号,是星期几
2012/06/01 Javascript
jquery得到font-size属性值实现代码
2013/09/30 Javascript
百度判断手机终端并自动跳转js代码及使用实例
2014/06/11 Javascript
基于Bootstrap+jQuery.validate实现Form表单验证
2014/12/16 Javascript
js+cookies实现悬浮购物车的方法
2015/05/25 Javascript
简要了解jQuery移动web开发的响应式布局设计
2015/12/04 Javascript
javascript每日必学之运算符
2016/02/16 Javascript
引用jquery框架后出错的解决方法
2016/08/09 Javascript
jQuery实现删除li节点的方法
2016/12/06 Javascript
javascript使用递归算法求两个数字组合功能示例
2017/01/03 Javascript
angularJS模态框$modal实例代码
2017/05/27 Javascript
jQuery实现的别踩白块小游戏完整示例
2019/01/07 jQuery
基于javascript处理nginx请求过程详解
2020/07/07 Javascript
前端性能优化建议
2020/09/17 Javascript
[02:10]DOTA2亚洲邀请赛 EG战队出场宣传片
2015/02/07 DOTA
[01:58]2018DOTA2亚洲邀请赛趣味视频——交流
2018/04/03 DOTA
全面了解python中的类,对象,方法,属性
2016/09/11 Python
flask中的wtforms使用方法
2018/07/21 Python
详解安装mitmproxy以及遇到的坑和简单用法
2019/01/21 Python
CSS3色彩模式有哪些?CSS3 HSL色彩模式的定义
2016/04/26 HTML / CSS
高清屏中使用Canvas绘图出现模糊的问题及解决方法
2019/06/03 HTML / CSS
花园仓库建筑:Garden Buildings Direct
2018/02/16 全球购物
Chemist Warehouse中文网:澳洲连锁大药房
2021/02/05 全球购物
Linux文件操作命令都有哪些
2016/07/23 面试题
军训鉴定表自我鉴定
2014/02/13 职场文书
幼儿教师工作感言
2014/02/14 职场文书
消防安全培训工作总结
2015/10/23 职场文书
MySQL选择合适的备份策略和备份工具
2022/06/01 MySQL
前端框架ECharts dataset对数据可视化的高级管理
2022/12/24 Javascript