iOS 简单工厂模式
iOS 简单工厂模式
什么是简单工厂模式?
简单工厂模式中定义一个抽象类,抽象类中声明公共的特征及属性,抽象子类继承自抽象类,去实现具体的操作。工厂类根据外界需求,在工厂类中创建对应的抽象子类实例并传给外界,而对象的创建是由外界决定的。外界只需要知道抽象子类对应的参数即可,而不需要知道抽象子类的创建过程,在外界使用时甚至不用引入抽象子类。
简单工厂模式将操作对象的创建,和关于操作对象相关的业务逻辑分离开,降低操作对象的耦合度。由于工厂类只是为外界创建对象,所以并不需要实例化工厂类对象,只需要为外界提供类方法即可。外界需要什么类型的抽象子类,只需要传递对应的参数即可。
简单工厂模式主要包含三部分:
- 工厂类:根据外界的需求,决定创建并返回哪个具体的抽象子类。
- 抽象类:定义抽象子类所需的属性和方法,子类通过继承自抽象类获得这些方法。
- 抽象子类:继承自抽象类,是具体操作的实现者。
业务场景
简单工厂模式主要适用于抽象子类的业务逻辑相同,但具体实现不同的情况。不同的操作子类执行同样的方法,最后的结果却是不同的,这也是多态的一种表现方式。
这里用一个简单的加减乘除的基础运算例子来说明一下,下面的UML类图和代码都会依据这个场景来实现。假设现在需要实现一个简单的加减乘除运算,这些运算具体操作都是类似的,都有两个被操作的值,只是运算符不同,这种情况就适合用简单工厂模式。
UML类图
根据上面提出的业务场景来画一张类图。
简单工厂模式
从上面图中我们可以看出,图中定义了一个运算抽象类,所有的运算操作类继承自这个运算抽象类。运算抽象类有两个参与运算的属性,通过调用getResult方法来获取这两个值最后运算的结果,调用方式都一样,只是最后的结果不同。抽象类并不参与运算,运算的结果通过运算操作类重载getResult方法去实现。
上图中还定义了一个简单工厂类,这个简单工厂类就是用于实现运算操作类实例化的逻辑,通过外界传进来的type参数,并将实例完成的运算操作类返回。
普通方式代码实现
首先定义抽象类,抽象类中将会包含参与运算的抽象子类的属性和行为(方法)。
//.h
@interface Operation : NSObject
@property (nonatomic, assign) CGFloat numberOne;
@property (nonatomic, assign) CGFloat numberTwo;
- (CGFloat)getResult;
@end
//.m
@implementation Operation
- (CGFloat)getResult {
return 0;
}
@end
定义抽象类之后,需要创建负责具体运算的抽象子类,也就是操作类,简单的定义了一下,代码不太多就全贴出来了。
//.h
@interface OperationAdd : Operation
@end //.m
@implementation OperationAdd
- (CGFloat)getResult {
return self.numberOne + self.numberTwo;
}
@end
//.h
@interface OperationSub : Operation
@end
//.m
@implementation OperationSub
- (CGFloat)getResult {
return self.numberOne - self.numberTwo;
}
@end
//.h
@interface OperationMul : Operation
@end
//.m
@implementation OperationMul
- (CGFloat)getResult {
return self.numberOne * self.numberTwo;
}
@end
//.h
@interface OperationDiv : Operation
@end
//.m
@implementation OperationDiv
- (CGFloat)getResult {
if (self.numberTwo == 0) {
NSLog(@"除数不能为零");
return 0;
} else {
return self.numberOne / self.numberTwo;
}
}
@end
下面先定义了四个静态变量,这四个静态变量声明了创建对象的类型,在后面反射部分代码中也会用到这四个静态变量。
static NSString kOperationAdd = @"OperationAdd";
static NSString kOperationSub = @"OperationSub";
static NSString kOperationMul = @"OperationMul";
static NSString kOperationDiv = @"OperationDiv";
现在具体参与运算的类都已经定义完成,就需要定义工厂类了。工厂类的职责就是根据外界需要,创建对应的抽象子类实例并返回给外界。
//.h
@interface OperationFactory : NSObject
+ (Operation )CreateOperationWithType:(NSString )type;
@end
//.m
@implementation OperationFactory
+ (Operation )CreateOperationWithType:(NSString )type {
if ([kOperationAdd isEqualToString:type]) {
return [OperationAdd new];
} else if ([kOperationSub isEqualToString:type]) {
return [OperationSub new];
} else if ([kOperationMul isEqualToString:type]) {
return [OperationMul new];
} else if ([kOperationDiv isEqualToString:type]) {
return [OperationDiv new];
}
return nil;
}
@end
上面我们就将工厂设计模式的定义都完成了,现在需要的就是外界直接拿来使用了。上面工厂类直接定义的类方法,因为外界获取某个具体的抽象子类时,并没有必要将工厂类实例化,工厂类只是完成一个功能。
-(void)viewDidLoad {
Operation *oper = [OperationFactory CreateOperationWithType:kOperationAdd];
oper.numberOne = 13;
oper.numberTwo = 24;
NSLog(@"result : %f", [oper getResult]);
}
到目前为止简单工厂模式的代码就写完了,可以看到外界想进行什么类型的运算,只需要将传入的运算类型参数改一下即可,工厂类就会实例化其他的抽象子类进行运算。但是这种工厂类的设计,有一个很大的问题,就在于每次增加或删除某个算法时,都需要对工厂类进行修改,这是不符合开放封闭原则的。对于这个问题,我们将会通过反射机制来进行处理。
工厂模式也是对面向对象编程三大特性之一的多态的一个很好的表述,下面将会简单的介绍一下多态的特性。
简单介绍一下多态
面向对象编程三大特性之一就有多态,多态是指在程序运行时,相同的消息可能会发给继承自同一个父类的不同类型的对象,虽然是同一个方法,但是运行时系统会根据当前对象所属的具体类型作出不同的响应。
面向对象三大特性中,继承和封装都是为了代码重用,继承可以继承自父类的特征和属性,封装可以将实现细节封装,外界调用实现某些功能。而多态则是为了接口重用。
多个子类继承同一个父类,就会具有和父类相同的行为和特征,子类可以对父类的方法进行重写,所以可能同一个方法每个子类的实现都不同。通过父类指针指向任意子类对象并调用相同方法,可能会得到不同的结果。
简单的说就是系统允许将当前类的指针,指向任何继承自当前类的子类,并且不会报错。由于子类继承自父类,所以和父类有相同的特征(方法)。当前类的指针向指向的子类对象发送消息,系统会根据具体的子类对父类方法的实现,作出不同的响应。 例如下面这行代码:
Operation *obj = [OperationFactory CreateOperationWithType:kOperationAdd];
在OC语言中大多数类都是继承自NSObject类的(也有继承自NSProxy等类),所以都具有NSObject的方法,如果子类对NSObject的方法进行了重写,NSObject指向子类对象并发送消息的结果就不一样了。
在上面这个例子中,OperationFactory工厂类将会返回Operation的子类实例,Operation的子类分别继承自同一父类,并且对其getResult方法进行了重写。Operation实例化的obj指针可能指向任何Operation的子类,并对其发送getResult消息。最终的结果会根据obj指针指向的子类有不同的结果,这就是多态。
我对多态的了解非常浅薄,有不对之处还请多多指出,这里只是顺带提了一下。
配合反射机制优化代码
在上面的代码中,我们会发现工厂类创建抽象子类的代码都是相同的,只是创建的具体对象不同,而且如果抽象子类很多的话,会有过多的条件语句。编程中这种重复代码我们都要将其简化,不然写出的代码就像垃圾代码一样,这也是新手和老手的区别之一。
在这里我们可以利用反射机制来简化代码,根据外面需要的操作子类的类型,反射出具体的类。在上面我们已经定义了一些NSString类型的静态变量,这些静态变量的值就是反射需要的字符串,外界只需要使用这些静态变量即可,不用自己手打字符串,也防止了错误的发生。修改之后外界不需要发生任何变化,只需要知道这些静态变量即可,只需要对工厂类进行修改。
只需将OperationFactory的创建方法改一下实现,其他地方不受影响。
- (Operation )CreateOperationWithType:(NSString )type { return [NSClassFromString(type) new]; }
改完之后的代码非常符合面向对象编程的开放封闭原则,即当外界需求发生变化时,只对现有代码进行扩展,不对其原有代码进行修改的原则。
现在假设再增加一个其他运算功能,只需要再创建一个继承自抽象类的抽象子类,在抽象子类中重写getResult方法来实现运算,并且在上面定义的静态变量中加入一个对应的变量。其他地方都不会受到影响,这就是一个比较好的面向对象的设计。
iOS 简单工厂模式的更多相关文章
- iOS开发-简单工厂模式
设计模式(Design pattern)是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.设计模式是为了可重用代码.让代码更容易被他人理解.保证代码可靠性.概念很长,iOS开发中最常 ...
- iOS设计模式 - (3)简单工厂模式
iOS设计模式 - (3)简单工厂模式 by Colin丶 转载请注明出处: http://blog.csdn.net/hitwhylz/article/ ...
- IOS设计模式浅析之简单工厂模式(SimpleFactory)
概述 首先说明一下,简单工厂模式不属于23种GOF设计模式之一.它也称作静态工厂方法模式,是工厂方法模式的特殊实现.这里对简单工厂模式进行介绍,是为本系列后面的工厂方法和抽象工厂模式做一个引子. 定义 ...
- iOS设计模式之简单工厂模式
简单工厂模式 基本理解 到底要实例化谁,将来会不会增加实例化的对象,比如计算器增加开根运算,这是很容易变化的地方,应该考虑用一个单独的类来做这个创造实例的过程,这就是工厂. 通过使用工厂模式,我们可以 ...
- iOS设计模式:简单工厂模式
1.简述 首先需要说明一下,简单工厂模式不属于23种GOF设计模式之一.它也称作静态工作方法模式,是工厂方法模式的特殊实现(也就是说工厂模式包含简单工厂模式).这里对简单工厂模式进行介绍,是为后面的工 ...
- iOS常用设计模式——工厂方法(简单工厂模式,工厂方法模式, 抽象工厂模式)
1. 简单工厂模式 如何理解简单工厂,工厂方法, 抽象工厂三种设计模式? 简单工厂方法包含:父类拥有共同基础接口,具体子类实现子类特殊功能,工厂类根据参数区分创建不同子类实例.该场景对应的UML图如下 ...
- iOS经常使用设计模式——工厂方法(简单工厂模式,工厂方法模式, 抽象工厂模式)
1. 简单工厂模式 怎样理解简单工厂,工厂方法. 抽象工厂三种设计模式? 简单工厂的生活场景.卖早点的小摊贩.他给你提供包子,馒头,地沟油烙的煎饼等,小贩是一个工厂.它生产包子,馒头,地沟油烙的煎饼. ...
- iOS 抽象工厂模式
iOS 抽象工厂模式 什么是抽象工厂模式 简单了解一下 按照惯例,我们先了解一下什么是抽象工厂模式.抽象工厂模式和工厂方法模式很相似,但是抽象工厂模式将抽象发挥的更加极致,是三种工厂模式中最抽象的一种 ...
- PHP设计模式(一)简单工厂模式 (Simple Factory For PHP)
最近天气变化无常,身为程序猿的寡人!~终究难耐天气的挑战,病倒了,果然,程序猿还需多保养自己的身体,有句话这么说:一生只有两件事能报复你:不够努力的辜负和过度消耗身体的后患.话不多说,开始吧. 一.什 ...
随机推荐
- smartjs - DataManager 场景示例分析 - 数据懒加载
发一张policy的参数图设置图: 场景1 - 数据的懒加载/延迟加载 在很多时候,为了提高网页的加载速度,减少不必要的开销,会将页面的数据拆分成几个部分,首先加载呈现可视区域内的数据,然后剩下来的会 ...
- Fidder模拟Post请求
背景 最近想用fidder模拟post请求,怎么都传值失败,发现写Composer => Request Body中写的内容,总是无法映射到mvc的action参数上.百度一番,发现如下解决方案 ...
- 代码演示 .NET 4.5 自带的 ReadonlyCollection 的使用
代码如下: 1. using System; using System.Collections.Generic; using System.Linq; using System.Text; using ...
- 自己动手搭建 Redis 环境,并建立一个 .NET HelloWorld 程序测试
关于 Redis ,下面来自百度百科: redis是一个key-value存储系统.和Memcached类似,它支持存储的value类型相对更多,包括string(字符串).list(链表).set( ...
- [IR] Information Extraction
阶段性总结 Boolean retrieval 单词搜索 [Qword1 and Qword2] O(x+y) [Qword1 and Qword2]- 改进: Gallo ...
- Hadoop第9周练习—Hive部署测试(含MySql部署)
1.1 2 :搭建Hive环境 内容 2.2 3 运行环境说明 1.1 硬软件环境 线程,主频2.2G,6G内存 l 虚拟软件:VMware® Workstation 9.0.0 build-8 ...
- BackgroundCheck – 根据图片亮度智能切换元素样式
BackgroundCheck 是一个轻量的 JavaScript 库,能够根据元素后面的图片的亮度自动切换元素样式.例如在图片幻灯片功能中,根据图片亮度调整导航箭头的颜色,这样让图片和导航的颜色形成 ...
- sql server service broker中调用存储过程执行跨库操作,不管怎么设置都一直提示 服务器主体 "sa" 无法在当前安全上下文下访问数据库 "dbname"。
用sql server自带的消息队列service borker,调用存储过程中,执行了一个跨库的操作,先是用了一个用户,权限什么都给够了,但是一直提示 服务器主体 "user" ...
- JavaScript 中变量、作用域和内存问题的学习
这是我学习JavaScript的第二篇文章,之前做过几年的Java开发,发现JavaScript虽然也是面向对象的语言但是确实有很多不同之处.就本篇博客,主要学习总结一下最近学习到的JavaScrip ...
- JavaScript中经典方法
jQuery()通过name名称获取当前name中value数组 /** 获取input中name属性相同的 value数组 */ function my_array_name(m){ var val ...