JavaScript设计模式(3)-工厂模式
工厂模式
1. 简单工厂
简单工厂:使用一个类或对象封装实例化操作
假如我们有个自行车商店类 BicycleShop,它提供了销售自行车的方法可以选择销售两类自行车 Speedster,ComfortCruiser,那么确定要销售那种自行车就可以通过简单工厂来实现。
注意:后面的代码中会用到 Javascript设计模式(1) 里面讲到的接口类 Interface 和 继承函数 extend
var Speedster = function() {}
Speedster.prototype = {
assemble: function() {}
}
var ComfortCruiser = function() {}
ComfortCruiser.prototype = {
assemble: function() {}
}
// 简单工厂模式
var BicycleFactory = {
createBicycle: function(model) {
var bicycle;
switch(model) {
case 'The Speedster':
bicycle = new Speedster();
break;
case 'The Comfort Cruiser':
default:
bicycle = new ComfortCruiser()
}
//检查是否实现了相应父类的方法
// Interface.ensureImplement(bicycle, Bicycle)
return bicycle
}
}
var BicycleShop = function() {};
BicycleShop.prototype = {
sellBicycle: function(model) {
var bicycle = BicycleFactory.createBicycle(model); // 通过简单工厂来实例化成员对象
bicycle.assemble()
return bicycle
}
}
// usage
var shop = new BicycleShop()
var b1 = shop.sellBicycle('aaaa')
// 可提供车型的所有信息都集中在一个地方管理
但是这里有个问题,就是所有的 BicycleShop 实例是能销售简单工厂对象里面的自行车,而现实生活中,不同的自行车商店可能会销售不同的自行车,而且这样实现会使得 BicycleFactory 这个简单工厂对象里面的 createBicycle 方法太大,试想像一下,全世界有多少种自信车,这里就会有多少个 case 分支。所以像这种销售自行车的场景,工厂模式会更适用
2. 工厂模式
这里的工厂指的是一个将其成员对象的实例化推迟到子类中进行的类
如果在不同时间或环境中实例的成员变量可能是不同实例时,我们一般就会用到工厂模式来处理这个实例化操作,一般会在类中会指定用以实例化成员变量的工厂方法 createXXX 供类的其他方法调用,所以这些方法就能通过统一的接口获取一个拥有已知接口的对象而不需要关系它是什么实现的,然后在该类的实例中会具体实现这个方法。
var AcmeSpeedster = function() {}
AcmeSpeedster.prototype = {
assemble: function() {}
}
var AcmecomfortCruiser = function() {}
AcmecomfortCruiser.prototype = {
assemble: function() {}
}
// BicycleShop 就是一个工厂,它是一个抽象类,需要在继承后实现里面用于实例化的工厂方法
var BicycleShop = function() {};
BicycleShop.prototype = {
sellBicycle: function(model) {
var bicycle = this.createBicycle(model)
bicycle.assemble()
return bicycle;
},
// 工厂方法
createBicycle: function(model) {
throw new Error('Unsupported operation on an abstract class.')
}
}
var AcmeBicycleShop = function() {};
extend(AcmeBicycleShop, BicycleShop);
AcmeBicycleShop.prototype.createBicycle = function(model) {
var bicycle;
switch(model) {
case 'The Speedster':
bicycle = new AcmeSpeedster();
break;
case 'The Comfort Cruiser':
default:
bicycle = new AcmecomfortCruiser();
}
// Interface.ensureImplement(bicycle, Bicycle)
return bicycle
}
// usage
var ashop = new AcmeBicycleShop()
ashop.sellBicycle()
// 对 Bycycle 进行一般性操作的代码可以全部写在父类 BicycleShop 中,而具体的 Bicycle 对象进行实例化的工作则别留到子类中。
3. 示例:XHR工厂
下面使用工厂模式封装了一个简单的 Ajax API,它可以兼容各大浏览器获取 xhr 对象,并提供了发送请求的 require 方法
var SimpleHandle = function() {};
SimpleHandle.prototype = {
require: function(method, url, callback, postVars) {
var xhr = this.createXhrObject();
xhr.onreadystatechange = function() {
if(xhr.readyState !== 4) return;
(xhr.status === 200) ?
callback.success(xhr.responseText, xhr.responseXML):
callback.failure(xhr.status);
}
xhr.open(method, url, true);
if(method != 'POST') postVars = null;
xhr.send(postVars)
},
createXhrObject: function() {
var methods = [
function() { return new XMLHttpRequest() },
function() { return new ActiveXObject('Msxml2.XMLHTTP') },
function() { return new ActiveXObject('Microsoft.XMLHTTP') }
];
for(var i=0, len=methods.length; i<len; i++) {
try {
methods[i]();
} catch(err) {
continue;
}
// memoizing : 返回所创建的对象并将自身改为用以创建那个对象的函数
this.createXhrObject = methods[i];
return methods[i]()
}
throw new Error('SimpleHandle: Could not create an XHR object.')
}
}
// usage
var myHandler = new SimpleHandle();
var callback = {
success: function(responseText) {alert('Success: ' + responseText)},
failure: function(statusCode) {alert('Failure: ' + statusCode)}
}
myHandler.require('POST', 'http://www.baidu.com', callback)
4. 示例 :RSS阅读器
这个例子中会用到上面的 XHR工厂 例子中实现的 Ajax API 来实现一个 RSS 阅读器。
// 这里用到了工厂模式
var SimpleHandle = function() {};
SimpleHandle.prototype = {
require: function(method, url, callback, postVars) {
var xhr = this.createXhrObject();
xhr.onreadystatechange = function() {
if(xhr.readyState !== 4) return;
(xhr.status === 200) ?
callback.success(xhr.responseText, xhr.responseXML):
callback.failure(xhr.status);
}
xhr.open(method, url, true);
if(method != 'POST') postVars = null;
xhr.send(postVars)
},
createXhrObject: function() {
var methods = [
function() { return new XMLHttpRequest() },
function() { return new ActiveXObject('Msxml2.XMLHTTP') },
function() { return new ActiveXObject('Microsoft.XMLHTTP') }
];
for(var i=0, len=methods.length; i<len; i++) {
try {
methods[i]();
} catch(err) {
continue;
}
// memoizing : 返回所创建的对象并将自身改为用以创建那个对象的函数
this.createXhrObject = methods[i]
return methods[i]()
}
throw new Error('SimpleHandle: Could not create an XHR object.')
}
}
// 这里是一个呈现数据的类
var ListDisplay = function(id, parent) {
this.list = document.createElement('ul');
this.list.id = id;
parent.appendChild(this.list)
}
ListDisplay.prototype = {
append: function(text) {
var newEl = document.createElement('li');
this.list.appendChild(newEl);
newEl.innerHTML = text;
return newEl
},
remove: function(el) {
this.list.removeChild(el);
},
clear: function() {
this.list.innerHTML = '';
}
}
var FeedReader = function(display, xhrHandler, conf) {
this.display = display;
this.xhrHandler = xhrHandler;
this.conf = conf;
this.startUpdates();
}
FeedReader.prototype = {
fetchFeed: function() {
var that = this;
var callback = {
success: function(text, xml) {
that.parseFeed(text, xml)
},
failure: function(status) {
that.showError(status)
}
}
this.xhrHandler.require('GET', this.conf.feedUrl, callback);
},
parseFeed: function(responseText, responseXML) {
this.display.clear()
var items = responseXML.getElementsByTagName('item');
for(var i=0, len=items.length; i<len; i++) {
var title = items[i].getElementsByTagName('title')[0];
var link = items[i].getElementsByTagName('link')[0];
this.display.append('<a href="' + link.firstChild.data + '">'+ title.firstChild.data + '</a>' )
}
},
showError: function(statusCode) {
this.display.clear();
this.display.append('Error fetching feed.')
},
stopUpdates: function() {
clearInterval(this.interval);
},
startUpdates: function() {
this.fetchFeed();
var that = this;
this.interval = setInterval(function() {
that.fetchFeed();
}, this.conf.updateInterval * 1000)
}
}
// 这里用到了前面讲过的单利对象,里面有一个 createFeedReader 方法用来将多个小对象组成一个大对象,也是一种工厂模式的使用
var FeedManager = {
createFeedReader: function(conf) {
var displayModule = new ListDisplay(conf.id + '-display', conf.parent)
// Interface.ensureImplements(displayModule, DisplayModule)
var xhrHandler = new SimpleHandle()
// Interface.ensureImplements(xhrHandler, AjaxHandler)
return new FeedReader(displayModule, xhrHandler, conf)
}
}
var conf = {
id: 'cnn-top-stories',
feedUrl: 'https://laike9m.com/blog/rss/', //http://www.adaymag.com/feed/',
updateInterval: 60,
parent: document.getElementById('feed-readers')
}
// usage
FeedManager.createFeedReader(conf)
5. 工厂模式的适用场合
- 动态实现
- 创建一些用不同方法实现统一接口的对象(如上面自行车例子)
- 可明确的实现统一接口【自行车】,也可隐含的实现(根据网速等环境原因) 【XHR 工厂】
- 节省设置开销
- 把实例化对象时重复的操作集中在一个地方
- 用许多小型对象组成一个大对象 【FeedManager】
6. 工厂模式之弊
- 如果不会另外更换一个类,或者不需要在运行期间在一系列可互换的类中进行选择,那就不应该使用工厂方法
- 使用 new 实例化比通过工厂模式实例化更简单易懂
- 切勿滥用,如果拿不定主意,那就不要用,因为在重构代码的时候还有机会使用
注意
转载、引用,但请标明作者和原文地址
JavaScript设计模式(3)-工厂模式的更多相关文章
- JavaScript设计模式之工厂模式
一.工厂模式概念 工厂模式定义一个用于创建对象的接口,这个接口由子类决定实例化哪一个类.该模式使一个类的实例化延迟到了子类.而子类可以重写接口方法以便创建的时候指定自己的对象类型(抽象工厂). 这个模 ...
- JavaScript设计模式--简单工厂模式
一,介绍 工厂模式创建对象(视为工厂里的产品)时无需指定创建对象的具体类. 工厂模式定义一个用于创建对象的接口,这个接口由子类决定实例化哪一个类.该模式使一个类的实例化延迟到了子类.而子类可以重写接口 ...
- javaScript设计模式之----工厂模式
什么是工厂模式?我们通过一个例子了解一下: 比如我们想要弹出几个字符串 function funA(){ alert('a'); } function funB(){ alert('b'); } fu ...
- JavaScript设计模式-10.工厂模式实例xhr
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- 【javascript】javascript设计模式之工厂模式
1.要解决的问题 2.如何实现 3.与构造函数的区别 4.总结 1.要解决的问题 工厂模式通常用于重复创建相似对象,提供动态创建对象的接口. 2.工厂模式最为设计模式中构造模式之一,通常在类或类的静态 ...
- javascript设计模式-抽象工厂模式
<!doctype html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- JavaScript设计模式-9.工厂模式
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- JavaScript设计模式--简单工厂模式例子---XHR工厂
第一步,Ajax操作接口(目的是起一个接口检测作用) (1)引入接口文件 //定义一个静态方法来实现接口与实现类的直接检验 //静态方法不要写出Interface.prototype ,因为这是写到接 ...
- JavaScript 设计模式之工厂模式
随机推荐
- BZOJ 3782: 上学路线 [Lucas定理 DP]
3782: 上学路线 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 192 Solved: 75[Submit][Status][Discuss] ...
- SDP(6):分布式数据库运算环境- Cassandra-Engine
现代信息系统应该是避不开大数据处理的.作为一个通用的系统集成工具也必须具备大数据存储和读取能力.cassandra是一种分布式的数据库,具备了分布式数据库高可用性(high-availability) ...
- ajax请求 readyState为0 可能原因之一
问题:同样的代码逻辑,PC端和iOS都能正常访问,但是Android系统请求都是报错: 上网查阅,关于ajax请求失败且状态码都是0的情况有很多,最后排查的原因是:域名证书问题:
- shell编程值之正则表达式与字符截取(6)
正则表达式与通配符 正则表达式用来在文件中匹配符合条件的字符串,正则是包含匹配.grep.awk.sed等命令可以支持正则表达式 通配符用来匹配符合条件的文件名,通配符是完全匹配.ls.find.cp ...
- MarkDown思考
前言 使用MarkDown有一段时间了,感觉的确很方便.大大提高了工作效率,并带来了良好的工作体验.但是,一直以来我都有一个困惑,就是MarkDown的插件和编辑器纷繁无比,却通常各自有一套自己的实现 ...
- mysql查找以逗号分隔的值-find_in_set
有了FIND_IN_SET这个函数.我们可以设计一个如:一只手机即是智能机,又是Andriod系统的. 比如:有个产品表里有一个type字段,他存储的是产品(手机)类型,有 1.智能机,2.Andri ...
- Rsync(远程同步): linux中Rsync命令的实际示例
rsync的 ( 远程同步 )为在Linux / Unix系统局部 拷贝和同步文件和目录远程以及一个最常用的命令. 随着rsync命令的帮助,您可以复制并在目录中远程和本地同步数据,在磁盘和网络,进行 ...
- How to install tcpping on Linux.md
To install tcptraceroute on Debian/Ubuntu: $ sudo apt-get install tcptraceroute To install tcptracer ...
- [Uva10294]Arif in Dhaka
[Uva10294]Arif in Dhaka 标签: 置换 Burnside引理 题目链接 题意 有很多个珠子穿成环形首饰,手镯可以翻转和旋转,项链只能旋转.(翻转过的手镯相同,而项链不同) 有n个 ...
- MYSQL 基础总结
学习笔记 [mysql 是不区分大小写的,要区分可以用相应的函数:所有标点符号全是英文状态下的] 一.基础部分 //创建数据库 Create database database_name; //使 ...