JavaScript设计模式读书笔记之一:接口
接口
在JavaScrip中模仿接口
用注释描述接口
/*
interface Composite {
function add(child);
function remove(child);
function getChild(index);
}
interface FormItem {
function save();
}
*/
var Composite = function(id, method, action) { // implements Composite, FormItem
};
Composite.prototype.add = function(child) {
console.log("add");
}
Composite.prototype.remove = function(child) {
console.log("remove");
}
Composite.prototype.getChild = function(index) {
console.log("getChild");
}
Composite.prototype.save = function() {
console.log("save");
};
var instance = new Composite(1, 2, 3);
instance.add();
instance.remove();
instance.getChild();
instance.save();
用属性检查模仿接口
概括:
- 一个方法用于表明自己实现了哪些接口。
- 一个方法用于验证是否实现了所有的接口,如果没有,则抛出异常。
- 一个方法用于判断某个类是否实现了所有的接口。
缺点:
- 自己说自己实现了什么接口,但是可能自己并没有实现,但是代码却不会抛出异常。
- 用了两层for循环,会引起性能消耗。
/*
interface Composite {
function add(child);
function remove(child);
function getChild(index);
}
interface FormItem {
function save();
}
*/
var CompositeForm = function(id, method, action) {
this.implementsInterface = ['Composite', 'FormItem'];
// ...
}
function addForm(instance) {
if(!implements(instance, 'Composite', 'FormItem')) {
throw new Error('Object does not implement a required interface');
}
if(!implements(instance, 'test', 'FormItem')) {
throw new Error('Object does not implement a required interface');
}
}
function implements(object) {
for(var i = 1;i < arguments.length;i++) {
var interfaceName = arguments[i];
var interfaceFound = false;
for(var j = 0;j < object.implementsInterface.length;j++) {
if(object.implementsInterface[j] == interfaceName) {
interfaceFound = true;
break;
}
}
if(!interfaceFound) {
return false;
}
}
return true;
}
var instance = new CompositeForm(1, 2, 3);
addForm(instance);
用鸭式辨型模仿接口
- 类是否声明自己支持哪些接口并不重要, 只要具有这些接口中的方法就可以了。
- 它把对象实现的方法集作为判断它是不是某个类的实例的唯一标准。
- 也就是说,如果对象具有与接口定义的方法同名的所有方法,那么就可以认为它实现了这个接口。
// Interface.js文件
/*
* 一个接口
* @param {string} name 接口的名字
* @param {array} methods 接口要实现的方法
* @class
*/
var Interface = function(name, methods) {
if(arguments.length !== 2) {
throw new Error("Interface constructor called with " + arguments.length
+ "arguments, but expectly exactly 2.");
}
this.name = name;
this.methods = [];
for(var i = 1;i < methods.length;i++) {
if(typeof methods[i] !== "string") {
throw new Error("Interface constructor expects method names to be "
+ "passed in as a string");
}
this.methods.push(methods[i]);
}
};
/*
* static class method 用于实现检测对象是否实现了接口里的所有方法
*
* @param {object} object 它应该包含一个对象的实例和至少一个接口
*
*/
Interface.prototype.ensureImplements = function(object) {
if(arguments.length < 2) {
throw new Error("Function Interface.ensureImplements called with " + arguments.length
+ "arguments, but expectly at least 2.");
}
for(var i = 1;i < arguments.length;i++) {
var interface = arguments[i];
if(interface.constructor !== Interface) {
throw new Error("Function interface.ensureImplements expects arguments "
+ "two and above to be instances of Interface.");
}
for(var j = 0, methodsLen = interface.methods.length;j < methodsLen;j++) {
var method = interface.methods[j];
if(!object[method] || typeof object[method] !== 'function') {
throw new Error("Function Interface.ensureImplements: Object "
+ "does not implement the " + interface.name
+ " interface.Method " + method + " was not found");
}
}
}
}
// index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title></title>
</head>
<body>
<script src="Interface.js"></script>
<script type="text/javascript">
// 实例化一个接口
var PeopleInterface = new Interface('PeopleInterface', ['run', 'eat']);
// 封装一个People类
var People = function(name) {
this.name = name;
};
People.prototype.run = function() {
console.log(this.name + " is running...");
};
People.prototype.eat = function() {
console.log(this.name + " is eating...");
};
// 实例化一个People对象
var yzf = new People('yzf');
function addPeople(instance) {
// 对yzf进行检测,看它是否完全继承了PeopleInterface接口
PeopleInterface.ensureImplements(yzf, PeopleInterface);
// 如果没有抛出异常,就可以使用下面这些方法了
yzf.run();
yzf.eat();
}
addPeople(yzf);
</script>
</body>
</html>
依赖于接口的设计模式
工厂模式
组合模式
装饰者模式
命令模式
JavaScript设计模式读书笔记之一:接口的更多相关文章
- JavaScript设计模式:读书笔记(未完)
该篇随我读书的进度持续更新阅读书目:<JavaScript设计模式> 2016/3/30 2016/3/31 2016/4/8 2016/3/30: 模式是一种可复用的解决方案,可用于解决 ...
- JavaScript设计模式 -- 读书笔记
JavaScript设计模式 一. 设计模式 一个模式就是一个可重用的方案: 有效的解决方法.易重用.善于表达该解决方案: 未通过"模式特性"测试的模式称为模式原型: 三规则:适用 ...
- 《你不知道的javascript》读书笔记2
概述 放假读完了<你不知道的javascript>上篇,学到了很多东西,记录下来,供以后开发时参考,相信对其他人也有用. 这篇笔记是这本书的下半部分,上半部分请见<你不知道的java ...
- HeadFirst设计模式读书笔记--目录
HeadFirst设计模式读书笔记(1)-策略模式(Strategy Pattern) HeadFirst设计模式读书笔记(2)-观察者模式(Observer Pattern) HeadFirst设计 ...
- 《编写可维护的javascript》读书笔记(中)——编程实践
上篇读书笔记系列之:<编写可维护的javascript>读书笔记(上) 上篇说的是编程风格,记录的都是最重要的点,不讲废话,写的比较简洁,而本篇将加入一些实例,因为那样比较容易说明问题. ...
- JavaScript设计模式学习笔记
1 JavaScript设计模式深入分析 私有属性和方法:函数有作用域,在函数内用var 关键字声明的变量在外部无法访问,私有属性和方法本质就是你希望在对象外部无法访问的变量. 特权属性和方法:创建属 ...
- Javascript & JQuery读书笔记
Hi All, 分享一下我学JS & JQuery的读书笔记: JS的3个不足:复杂的文档对象模型(DOM),不一致的浏览器的实现和便捷的开发,调试工具的缺乏. Jquery的选择器 a. 基 ...
- Head First 设计模式读书笔记(1)-策略模式
一.策略模式的定义 策略模式定义了算法族,分别封装起来,让它们之间可以互换替换,此模式让算法的变化独立使用算法的客户. 二.使用策略模式的一个例子 2.1引出问题 某公司做了一套模拟鸭子的游戏:该游戏 ...
- JavaScript设计模式系列学习笔记目录
说明 本系列笔记参考书籍<JavaScript设计模式>.<JavaScript高级程序设计3> 参考博客:汤姆大叔博客:http://www.cnblogs.com/TomX ...
随机推荐
- Pro Git CN Plus
Git — The stupid content tracker, 傻瓜内容跟踪器.Linus 是这样给我们介绍 Git 的. Git 是用于 Linux 内核开发的版本控制工具.与常用的版本控制工具 ...
- 关注云端搜索技术:elasticsearch,nutch,hadoop,nosql,mongodb,hbase,cassandra 及Hadoop优化
http://www.searchtech.pro/ Hadoop添加或调整的参数: 一.hadoop-env.sh1.hadoop的heapsize的设置,默认1000 # The maximum ...
- Laravel生命周期
如果你对一件工具的使用原理了如指掌,那么你在用这件工具的时候会充满信心! 一旦用户(浏览器)发送了一个HTTP请求,我们的apache或者nginx一般都转到index.php,因此,之后的一系列步骤 ...
- 程序员的自我修养:高效使用Google解决问题
如果票选近二十年最伟大的发明,我相信搜索引擎肯定会占据一个不容小觑的位置,它不单是一项发明,更是一项成就,最大程度消灭了信息的不平等.既然人人都可以接触到海量的信息,那么衡量信息财富多寡就只剩下技巧这 ...
- jquery小测
1.在div元素中,包含了一个<span>元素,通过has选择器获取<div>元素中的<span>元素的语法是? 提示使用has() $("div:has ...
- 《深入理解Java虚拟机》学习笔记之内存回收
垃圾收集(Garbage Collection,GC)并不是Java语言的半生产物,事实上GC历史远比Java久远,真正使用内存动态分配和垃圾收集技术的语言是诞生于1960年的Lisp语言.经过半个世 ...
- hdoj 1257 DP||贪心
最少拦截系统 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Subm ...
- 从覆盖bootstrap样式谈css选择器优先级
样式优先级 首先简单说几个定义样式的方式: 元素内嵌: <li><a href="" style="color:#ffffff;">SH ...
- Search a 2D Matrix leetcode
Write an efficient algorithm that searches for a value in an m x n matrix. This matrix has the follo ...
- C++类对象大小的计算
(一)常规类大小计算 C++类对象计算需要考虑很多东西,如成员变量大小,内存对齐,是否有虚函数,是否有虚继承等.接下来,我将对此举例说明. 以下内存测试环境为Win7+VS2012,操作系统为32位 ...