设计模式一(工厂模式)

工厂模式可以简单概括为同类型不同型号的产品有各自对应的工厂进行生产。
比如iPhone手机旗下有iPhoneX及iPhone8两种型号的手机,iPhoneX有自己iPhoneX的专属工厂进行生产,而iPhone8有自己iPhone8的专属工厂进行生产。两条生产线没有交集互不影响,后期如果新增或废弃相关类型的产品也不会影响到原有的生产线。

三种模式

  • 简单工厂模式:简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例。简单工厂模式是工厂模式家族中最简单实用的模式,可以理解为是不同工厂模式的一个特殊实现。

  • 工厂模式:抽象了工厂接口的具体产品,应用程序的调用不同工厂创建不同产品对象。(抽象产品)

  • 抽象工厂模式:在工厂模式的基础上抽象了工厂,应用程序调用抽象的工厂发发创建不同产品对象。(抽象产品+抽象工厂)

一、简单工厂模式

1、简单工厂模式定义

​ 简单工厂模式(Simple Factory Pattern):定义一个工厂类,它可以根据参数的不同返回不同类的实例,被创建的实例通常都具有共同的父类。因为在简单工厂模式中用于创建实例的方法是静态(static)方法,因此简单工厂模式又被称为静态工厂方法(Static Factory Method)模式,它属于类创建型模式。

​ 注意:简单工厂模式并不属于GoF 23个经典设计模式,但通常将它作为学习其他工厂模式的基础。

简单工厂模式的要点:当你需要什么,只需要传入一个正确的参数,就可以获取你所需要的对象,而无须知道其创建细节。

2、简单工厂模式包含的角色
  • Factory(工厂角色):工厂角色即工厂类,它是简单工厂模式的核心,负责实现创建所有产品实例的内部逻辑;工厂类可以被外界直接调用,创建所需的产品对象;在工厂类中提供了静态的工厂方法factoryMethod(),它的返回类型为抽象产品类型Product。

  • Product(抽象产品角色):它是工厂类所创建的所有对象的父类,封装了各种产品对象的公有方法,它的引入将提高系统的灵活性,使得在工厂类中只需定义一个通用的工厂方法,因为所有创建的具体产品对象都是其子类对象。

  • ConcreteProduct(具体产品角色):它是简单工厂模式的创建目标,所有被创建的对象都充当这个角色的某个具体类的实例。每一个具体产品角色都继承了抽象产品角色,需要实现在抽象产品中声明的抽象方法。

​ 简单工厂模式的要点:当你需要什么,只需要传入一个正确的参数,就可以获取你所需要的对象,而无须知道其创建细节。

3、结构图
简单工厂模式

父类Fruits具有一个属性name和一个方法sweet、poorTaste,AppleFruit和OrangeFruit继承自Fruits,简单工厂类为FruitFactory,负责根据传入的不同参数FruitType来决定生产的是AppleFruit还是OrangeFruit。

4、示例

本Demo还是以小贩卖苹果和橘子为例:(小贩就是一个工厂,他为你提供苹果,橘子等水果)

  • 先创建Fruits水果类,有一个sweet()、poorTaste()抽象方法,表示抽象产品类;
  • 然后创建AppleFruit和OrangeFruit类,继承自Fruits类,表示具体产品类;
  • 然后创建FruitFactory工厂类,有一个fruitsFactory:type方法,声明但未实现,表示抽象工厂;

具体代码如下:
Fruits类:

#import <Foundation/Foundation.h>  //Fruits.h

@interface Fruits : NSObject
/**水果名*/
@property (nonatomic,copy) NSString *name;
- (void)sweet; /**< 甜 */
- (void)poorTaste; /**< 不好吃 */
@end


#import "Fruits.h"     //Fruits.m

@implementation Fruits
// 甜
- (void)sweet {}

// 不好吃
- (void)poorTaste {}
@end

AppleFruit和OrangeFruit类

/*------------------AppleFruit------------------*/
#import "Fruits.h"    //AppleFruit.h

//遵循了开闭原则
@interface AppleFruit : Fruits
- (void)freshApple; /**< 新鲜的苹果 */   // 开闭原则
@end



#import "AppleFruit.h"       //AppleFruit.m

@implementation AppleFruit
@synthesize name;
// 甜
- (void)sweet {
    NSLog(@"%@非常甜的啦-----%s",self.name, __func__);
}

// 不好吃
- (void)poorTaste {
    NSLog(@"%@不好吃的啦-----%s",self.name,__func__);
}

// 新鲜的苹果
- (void)freshApple {
     NSLog(@"%@是新鲜的-----%s",self.name,__func__);
}
@end


/*------------------OrangeFruit------------------*/
#import "Fruits.h"                       //OrangeFruit.h

@interface OrangeFruit : Fruits

- (void)acidOrange; /**< 酸橘子 */
@end



#import "OrangeFruit.h"           //OrangeFruit.m

@implementation OrangeFruit
@synthesize name;
// 甜
- (void)sweet {
   NSLog(@"%@非常甜的啦-----%s",self.name, __func__);
}

// 不好吃
- (void)poorTaste {
    NSLog(@"%@不好的啦-----%s",self.name, __func__);
}

/**< 酸橘子 */
- (void)acidOrange {
    NSLog(@"%@有点酸-----%s",self.name, __func__);
}
@end

FruitFactory类:

#import <Foundation/Foundation.h>             //FruitFactory.h
#import "Fruits.h"
#import "AppleFruit.h"
#import "OrangeFruit.h"

typedef NS_ENUM(NSInteger,FruitType){
    FruitTypeApple,
    FruitTypeOrange,
};

@interface FruitFactory : NSObject
// 创造水果的工厂
+ (Fruits *)fruitsFactory:(FruitType)type;
@end



#import "FruitFactory.h"                         //FruitFactory.m
@implementation FruitFactory
+ (Fruits *)fruitsFactory:(FruitType)type {
    // 创建空的对象.在工厂方法里面进行水果的制造
    Fruits *fruits = nil;

    switch (type) {
        case FruitTypeApple:
            fruits = [[AppleFruit alloc] init];
            break;

        case FruitTypeOrange:
            fruits = [[OrangeFruit alloc] init];
            break;

        default:
            break;
    }
    return fruits;
}
@end

运行代码:

#import "ViewController.h"
#import "FruitFactory.h"
@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
   
/*###############SimpleFactoryPattern(简单工厂模式)######################*/
    // 在水果工厂里面创建出苹果
    Fruits *fruit =  [FruitFactory fruitsFactory:FruitTypeApple];
    fruit.name = @"苹果";
    [fruit sweet];
    
    // 在水果工厂里面创建出苹果
    Fruits *fruit1 =  [FruitFactory fruitsFactory:FruitTypeOrange];
    fruit1.name = @"橘子";
    [fruit1 poorTaste];
    
    // 在水果工厂里面创建出苹果, 调用私有的方法
    AppleFruit *apple =  (AppleFruit*)[FruitFactory fruitsFactory:FruitTypeApple];
    apple.name = @"苹果";
    [apple freshApple];
    
    // 在水果工厂里面创建出橘子, 调用私有的方法
    OrangeFruit *orage =  (OrangeFruit *)[FruitFactory fruitsFactory:FruitTypeOrange];
    orage.name = @"橘子";
    [orage acidOrange];
}
@end

打印结果:

2019-12-10 16:07:55.643264+0800 设计模式[78854:2676436] 苹果非常甜的啦------[AppleFruit sweet]
2019-12-10 16:07:55.643415+0800 设计模式[78854:2676436] 橘子不好的啦------[OrangeFruit poorTaste]
2019-12-10 16:07:55.643524+0800 设计模式[78854:2676436] 苹果是新鲜的------[AppleFruit freshApple]
2019-12-10 16:07:55.643629+0800 设计模式[78854:2676436] 橘子有点酸------[OrangeFruit acidOrange]
5、总结

简单工厂模式的简化:有时候,为了简化简单工厂模式,我们可以将抽象产品类和工厂类合并,将静态工厂方法移至抽象产品类中。比如在Fruits类中添加一个方法:

Fruits类:

#import <Foundation/Foundation.h>

typedef NS_ENUM(NSInteger,FruitType){
    FruitTypeApple,
    FruitTypeOrange,
};

@interface Fruits : NSObject
/**水果名*/
@property (nonatomic,copy) NSString *name;
- (void)sweet; /**< 甜 */
- (void)poorTaste; /**< 不好吃 */

// 创造水果的工厂
+ (Fruits *)fruitsFactory:(FruitType)type;
@end


#import "Fruits.h"
#import "AppleFruit.h"
#import "OrangeFruit.h"
@implementation Fruits
// 甜
- (void)sweet {}

// 不好吃
- (void)poorTaste {}

+ (Fruits *)fruitsFactory:(FruitType)type {
    // 创建空的对象.在工厂方法里面进行水果的制造
    Fruits *fruits = nil;

    switch (type) {
        case FruitTypeApple:
            fruits = [[AppleFruit alloc]init];
            break;

        case FruitTypeOrange:
            fruits = [[OrangeFruit alloc]init];
            break;

        default:
            break;
    }
    return fruits;
}
@end

运行代码:

Fruits *fruit =  [Fruits fruitsFactory:FruitTypeApple];
fruit.name = @"苹果";
[fruit sweet];
    
AppleFruit *apple =  (AppleFruit*)[Fruits fruitsFactory:FruitTypeApple];
apple.name = @"苹果";
[apple freshApple];

​ 简单工厂模式提供了专门的工厂类用于创建对象,将对象的创建和对象的使用分离开,它作为一种最简单的工厂模式在软件开发中得到了较为广泛的应用。

1)优点
  • 工厂类包含必要的判断逻辑,可以决定在什么时候创建哪一个产品类的实例,客户端可以免除直接创建产品对象的职责,而仅仅“消费”产品,简单工厂模式实现了对象创建和使用的分离。

  • 客户端无须知道所创建的具体产品类的类名,只需要知道具体产品类所对应的参数即可,对于一些复杂的类名,通过简单工厂模式可以在一定程度减少使用者的记忆量。

  • 通过引入配置文件,可以在不修改任何客户端代码的情况下更换和增加新的具体产品类,在一定程度上提高了系统的灵活性。

2)缺点
  • 由于工厂类集中了所有产品的创建逻辑,职责过重,一旦不能正常工作,整个系统都要受到影响。

  • 系统扩展困难,一旦添加新产品就不得不修改工厂逻辑,在产品类型较多时,有可能造成工厂逻辑过于复杂,不利于系统的扩展和维护。

  • 简单工厂模式由于使用了静态工厂方法,造成工厂角色无法形成基于继承的等级结构。

3)适用场景
  • 工厂类负责创建的对象比较少,由于创建的对象较少,不会造成工厂方法中的业务逻辑太过复杂。

  • 客户端只知道传入工厂类的参数,对于如何创建对象并不关心。

暂时先写到这,后续有时间再更新...

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,053评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,527评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,779评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,685评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,699评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,609评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,989评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,654评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,890评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,634评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,716评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,394评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,976评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,950评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,191评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,849评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,458评论 2 342

推荐阅读更多精彩内容