詳解iOS應(yīng)用開發(fā)中使用設(shè)計(jì)模式中的抽象工廠模式
概述
我們知道簡(jiǎn)單工廠模式的優(yōu)點(diǎn)是去除了客戶端與具體產(chǎn)品的依賴,缺點(diǎn)是違反了“開放-關(guān)閉原則”;工廠方法模式克服了簡(jiǎn)單工廠模式的缺點(diǎn),將產(chǎn)品的創(chuàng)建工作放到具體的工廠類,每個(gè)工廠類負(fù)責(zé)生成一個(gè)產(chǎn)品。但是在實(shí)際應(yīng)用中,一個(gè)工廠類只創(chuàng)建單個(gè)產(chǎn)品的情況很少,一般一個(gè)工廠類會(huì)負(fù)責(zé)創(chuàng)建一系列相關(guān)的產(chǎn)品,如果我們要設(shè)計(jì)這樣的系統(tǒng),工廠方法模式顯然不能滿足應(yīng)用的需求,本章要介紹的抽象工廠模式,可以很好地解決一系列產(chǎn)品創(chuàng)建的問題。
定義
“提供一個(gè)創(chuàng)建一系列相關(guān)或相互依賴對(duì)象的接口,而無需指定它們具體的類?!?/p>
最初的定義出現(xiàn)于《設(shè)計(jì)模式》(Addison-Wesley,1994)。
結(jié)構(gòu)圖
先對(duì)上面結(jié)構(gòu)圖的幾個(gè)角色進(jìn)行說明:
AbstractFactory:抽象工廠接口,里面應(yīng)該包含所有產(chǎn)品創(chuàng)建的抽象方法;
ConcreteFactory1和ConcreteFactory2:具體的工廠,創(chuàng)建具有特定實(shí)現(xiàn)的產(chǎn)品對(duì)象;
AbstractProductA和AbstractProductB:抽象產(chǎn)品,它們可能有多種不同的實(shí)現(xiàn)方式;
ProductA1、ProductA2、ProductB1和ProductB2:具體的產(chǎn)品,是抽象產(chǎn)品的具體實(shí)現(xiàn)。
從結(jié)構(gòu)圖中可以看到,抽象工廠方法最大的好處是能夠很方便的變換產(chǎn)品系列(例如id<AbstractFactory> factory =[ [ConcreteFactory1 alloc] init],只需要將ConcreteFactory1換成ConcreteFactory2,就可以創(chuàng)建ProductA2和ProductB2)。另外,抽象工廠方法讓具體的創(chuàng)建實(shí)例過程與客戶端分離,客戶端是通過它們的抽象接口操作實(shí)例,產(chǎn)品的具體類名也被具體工廠的實(shí)現(xiàn)分離,不會(huì)出現(xiàn)在客戶代碼中(例如id<AbstractProductA> product = [factory createProductA],客戶端根本不知道具體的類名是ProductA1還是ProductA2)。
但是,抽象工廠方法也是存在缺點(diǎn)的,比如說現(xiàn)在我們要增加一個(gè)新的產(chǎn)品,首先,我們需要增加三個(gè)類:AbstractProductC、ProductC1、ProductC2;另外,我們還需要更改三個(gè)類:AbstractFactory、ConcreteFactory1、ConcreteFactory2,這樣,很明顯是違背“開放-關(guān)閉原則”。這也是可以理解的,沒有任何一個(gè)設(shè)計(jì)模式是完美沒有瑕疵的,這就好比世界上沒有打不敗的武功一樣。我們可以做的就是在實(shí)際的需求中,盡可能的將變化點(diǎn)進(jìn)行隔離,以達(dá)到變化發(fā)生的時(shí)候,對(duì)整個(gè)系統(tǒng)的影響最小,變化所帶來的變更和成本最低。
示例
先給大家看一下數(shù)據(jù)庫訪問的類結(jié)構(gòu)圖吧。
好的,簡(jiǎn)單分析一下上面這張類結(jié)構(gòu)圖,這張圖中有三個(gè)獨(dú)立的模塊兒,一個(gè)是IFactory接口,以不同數(shù)據(jù)庫為劃分原則對(duì)部門進(jìn)行抽象,一個(gè)是對(duì)訪問數(shù)據(jù)庫的不同部門,還有一個(gè)是對(duì)數(shù)據(jù)庫操作的人員進(jìn)行了抽象。類圖中沒有提到接下來需要給大家展示的兩個(gè)類,一個(gè)是User類,一個(gè)是Department類,因?yàn)檫@兩個(gè)類是對(duì)數(shù)據(jù)庫數(shù)據(jù)的封裝,和結(jié)構(gòu)并沒有直接關(guān)系,所以沒有顯示出來,在此說明一下,以免大家引起混亂。其實(shí),靜下心來細(xì)細(xì)的看,結(jié)構(gòu)還是蠻清晰的。
呵呵,下面還是老樣子,給大家展示一下代碼。
注意:本文所有代碼均在ARC環(huán)境下編譯通過。
User類接口
#import <Foundation/Foundation.h>
@interface User :NSObject
@property int *ID;
@property NSString *Name;
@end
User類實(shí)現(xiàn)
#import "User.h"
@implementation User
@synthesize Name =_Name;
@synthesize ID =_ID;
@end
Department類接口
#import <Foundation/Foundation.h>
@interface Department:NSObject
@property int *ID;
@property NSString *DeptName;
@end
Department類實(shí)現(xiàn)
#import "Department.h"
@implementation Department
@synthesize ID =_ID;
@synthesize DeptName =_DeptName;
@end
IDepartment類接口
#import <Foundation/Foundation.h>
@class Department;
@interface IDepartment :NSObject
-(void)Insert:(Department*)department;
-(Department*)GetDepartment:(int)myId;
@end
IDepartment類實(shí)現(xiàn)
#import "IDepartment.h"
#import "Department.h"
@implementation IDepartment
-(void)Insert:(Department *)department{
return;
}
-(Department*)GetDepartment:(int)myId{
return nil;
}
@end
SqlserverDepartment類接口
#import "IDepartment.h"
@interface SqlserverDepartment:IDepartment
@end
SqlserverDepartment類實(shí)現(xiàn)
#import "SqlserverDepartment.h"
@implementation SqlserverDepartment
-(void)Insert:(Department *)department{
NSLog(@"在SQL Server中給Department表增加一條記錄");
}
-(Department*)GetDepartment:(int)myId{
NSLog(@"在SQL Server中根據(jù)ID得到Department表一條記錄");
return nil;
}
@end
AccessDepartment類接口
#import "IDepartment.h"
@interface AccessDepartment:IDepartment
@end
*AccessDepartment類實(shí)現(xiàn)
#import "AccessDepartment.h"
@implementation AccessDepartment
-(void)Insert:(Department *)department{
NSLog(@"在Access中給Department表增加一條記錄");
}
-(Department*)GetDepartment:(int)myId{
NSLog(@"在Access中根據(jù)myId得到Department表一條記錄");
return nil;
}
@end
IUser類接口
#import <Foundation/Foundation.h>
@class User;
@interfaceIUser :NSObject
-(void)Insert:(User*)user;
-(User*)GetUser:(int)myID;
@end
IUser類實(shí)現(xiàn)
#import "IUser.h"
#import "User.h"
@implementation IUser
-(void)Insert:(User *)user{
return;
}
-(User*)GetUser:(int)myID{
return nil;
}
@end
SqlServerUser類接口
#import "IUser.h"
@interface SqlServerUser :IUser
@end
SqlServerUser類實(shí)現(xiàn)
#import "SqlServerUser.h"
@implementation SqlServerUser
-(void)Insert:(User *)user{
NSLog(@"在SQL Server中給User表增加一條記錄");
}
-(User*)GetUser:(int)myID{
NSLog(@"在SQL Server中根據(jù)myID得到User表一條記錄");
return nil;
}
@end
AccessUser類接口
#import "IUser.h"
@interface AccessUser :IUser
@end
AccessUser類實(shí)現(xiàn)
#import "AccessUser.h"
@implementation AccessUser
-(void)Insert:(User *)user{
NSLog(@"在Access中給User表增加一條記錄");
}
-(User*)GetUser:(int)myID{
NSLog(@"在Access中根據(jù)myID得到User表一條記錄");
return nil;
}
@end
IFactories類接口
#import "AccessUser.h"
@implementation AccessUser
-(void)Insert:(User *)user{
NSLog(@"在Access中給User表增加一條記錄");
}
-(User*)GetUser:(int)myID{
NSLog(@"在Access中根據(jù)myID得到User表一條記錄");
return nil;
}
@end
IFactories類實(shí)現(xiàn)
#import "IFactories.h"
#import "IUser.h"
#import "IDepartment.h"
@implementation IFactories
-(IUser*)CreateUser{
return nil;
}
-(IDepartment*)CreateDepartment{
return nil;
}
@end
AccessFactory類接口
#import "IFactories.h"
@interface AccessFactory :IFactories
@end
AccessFactory類實(shí)現(xiàn)
#import "AccessFactory.h"
#import "AccessUser.h"
#import "AccessDepartment.h"
@implementation AccessFactory
-(IUser*)CreateUser{
return [[AccessUser alloc]init];
}
-(IDepartment*)CreateDepartment{
return [[AccessDepartment alloc]init];
}
@end
SqlServerFactory類接口
#import "IFactories.h"
@interface SqlServerFactory :IFactories
@end
SqlServerFactory類實(shí)現(xiàn)
#import "SqlServerFactory.h"
#import "SqlServerUser.h"
#import "SqlserverDepartment.h"
@implementation SqlServerFactory
-(IUser*)CreateUser{
return [[SqlServerUser alloc]init];
}
-(IDepartment*)CreateDepartment{
return [[SqlserverDepartment alloc]init];
}
@end
Main方法調(diào)用
#import <Foundation/Foundation.h>
#import "User.h"
#import "Department.h"
#import "IFactories.h"
#import "AccessFactory.h"
#import "IUser.h"
#import "IDepartment.h"
int main (int argc,const char * argv[])
{
@autoreleasepool{
User *user = [[User alloc]init];
Department *dept = [[Department alloc]init];
IFactories *factories = [[AccessFactory alloc]init];
IUser *iu = [factories CreateUser];
[iu Insert:user];
[iu GetUser:1];
IDepartment *myId = [factories CreateDepartment];
[myId Insert:dept];
[myId GetDepartment:1];
}
return 0;
}
上面羅列了一堆代碼,其實(shí),羅列這些代碼的目的只有一個(gè),就是為了幫助像我一樣基礎(chǔ)不太好的同學(xué)盡快入門,有一個(gè)感性的認(rèn)識(shí),邁過第一道門檻。
- iOS App設(shè)計(jì)模式開發(fā)中對(duì)interpreter解釋器模式的運(yùn)用
- 詳解iOS App設(shè)計(jì)模式開發(fā)中對(duì)于享元模式的運(yùn)用
- iOS App設(shè)計(jì)模式開發(fā)中對(duì)建造者模式的運(yùn)用實(shí)例
- iOS App設(shè)計(jì)模式開發(fā)中對(duì)迭代器模式的使用示例
- iOS App的設(shè)計(jì)模式開發(fā)中對(duì)State狀態(tài)模式的運(yùn)用
- 解析iOS應(yīng)用開發(fā)中對(duì)設(shè)計(jì)模式中的抽象工廠模式的實(shí)現(xiàn)
- 實(shí)例解析設(shè)計(jì)模式中的外觀模式在iOS App開發(fā)中的運(yùn)用
- 設(shè)計(jì)模式開發(fā)中的備忘錄模式在iOS應(yīng)用開發(fā)中的運(yùn)用實(shí)例
- 深入解析設(shè)計(jì)模式中的裝飾器模式在iOS應(yīng)用開發(fā)中的實(shí)現(xiàn)
- 舉例講解設(shè)計(jì)模式中的原型模式在iOS應(yīng)用開發(fā)中的作用
相關(guān)文章
iOS中設(shè)置清除緩存功能的實(shí)現(xiàn)方法
清除緩存基本上都是在設(shè)置界面的某一個(gè)Cell,于是我們可以把清除緩存封裝在某一個(gè)自定義Cell中,現(xiàn)在位大家介紹一種最基礎(chǔ)的清除緩存的方法,感興趣的朋友一起看看吧2017-07-07一個(gè)iOS上的秒表小應(yīng)用的實(shí)現(xiàn)方法分享
這篇文章主要介紹了一個(gè)iOS上的秒表小應(yīng)用的實(shí)現(xiàn)方法分享,代碼基于傳統(tǒng)的Objective-C,需要的朋友可以參考下2015-10-10你應(yīng)該知道的tableViewCell行高計(jì)算處理
這篇文章主要給大家介紹了關(guān)于tableViewCell行高計(jì)算的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-12-12IOS開發(fā)代碼分享之獲取啟動(dòng)畫面圖片的string
本文是IOS開發(fā)代碼分享系列的第一篇文章,這里分享下獲取啟動(dòng)畫面圖片的string的代碼,本代碼支持 iPhone 6 以下. 支持 iPhone 及 iPad,非常實(shí)用,希望對(duì)大家有所幫助2014-09-09