美文网首页收藏
抽象工厂模式-iOS实践

抽象工厂模式-iOS实践

作者: 野码道人 | 来源:发表于2022-01-26 11:36 被阅读0次

    意图:
    提供一个接口,用来创建一组相关或者相互依赖的对象,而无需指定他们的具体类
    适用范围:
    提供一个产品类库,显示它们的接口,隐藏具体实现,使系统独立于产品的创建、组合、表示

    缺点:
    当有新增品类的时候,同时需要拓展抽象工厂和具体工厂
    解决方案:
    抽象工厂需要更抽象,不再关心产品的品类,具体工厂需要更具体,只提供一种产品的规格参数
    具体实现:
    通过反射机制运行时动态创建产品,具体工厂不再负责生产,而是提供产品的具体参数,抽象工厂通过产品参数反射出具体工厂,然后生产一个产品

    实践-app壳工程

    大型app基本都经历过从简单架构到复杂架构的演进,像支付宝、美团这种体量的app不拆分业务线简直就是灾难,本文以一个app为工厂,该工厂生产各种业务线,就拿美团来举例,有骑行、打车、电影、外卖业务线

    定义一个业务线工厂协议AbstractFactoryProtocol.h,该协议只描述各个业务线的通用行为,比如是否是单例

    @protocol AbstractFactoryProtocol <NSObject>
    
    @optional
    
    + (id)shareInstance;
    
    @end
    

    定义一个抽象工厂类AbstractFactory,用单例来描述,该类负责通过产品规格参数,反射出具体工厂,然后生产产品

    @interface AbstractFactory : NSObject
    
    + (instancetype)sharedFactory;
    
    - (id)createProduct:(Protocol *)protocol;
    
    @end
    

    AbstractFactory的实现内容如下

    static NSString *kProtocol = @"FactoryMethodProtocol";
    static NSString *kViewController = @"Page";
    
    @implementation AbstractFactory
    
    + (instancetype)sharedFactory {
        static id sharedFactory = nil;
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
            sharedFactory = [[self alloc] init];
        });
        return sharedFactory;
    }
    
    - (id)createProduct:(Protocol *)protocol {
        NSString *protocolName = NSStringFromProtocol(protocol);
        NSString *className = [protocolName stringByReplacingOccurrencesOfString:kProtocol withString:kViewController];
        Class implClass = NSClassFromString(className);
        if (!implClass) {
            @throw [NSException exceptionWithName:NSInternalInconsistencyException reason:[NSString stringWithFormat:@"%@ protocol does not have match class", protocolName] userInfo:nil];
        }
        if ([[implClass class] respondsToSelector:@selector(shareInstance)]) {
            return [[implClass class] shareInstance];
        }
        return [[implClass alloc] init];
    }
    
    @end
    

    定义一个具体工厂集合SpecificProducts.h

    #ifndef SpecificProducts_h
    #define SpecificProducts_h
    
    #import "BikeFactoryMethodProtocol.h"
    #import "CarFactoryMethodProtocol.h"
    #import "MovieFactoryMethodProtocol.h"
    #import "TakeoutFactoryMethodProtocol.h"
    
    #endif /* SpecificProducts_h */
    

    具体工厂继承抽象工厂协议

    @protocol BikeFactoryMethodProtocol <NSObject, AbstractFactoryProtocol>
    
    @end
    

    定义具体产品,骑行、打车、电影、外卖

    @interface BikePage : UIViewController
    
    @end
    
    @interface CarPage : UIViewController
    
    @end
    
    @interface MoviePage : UIViewController
    
    @end
    
    @interface TakeoutPage : UIViewController
    
    @end
    

    客户端只需要引用抽象工厂和具体工厂集合即可生产所有产品

    id<BikeFactoryMethodProtocol> item0 = [[AbstractFactory sharedFactory] createProduct:@protocol(BikeFactoryMethodProtocol)];
    
    id<CarFactoryMethodProtocol> item1 = [[AbstractFactory sharedFactory] createProduct:@protocol(CarFactoryMethodProtocol)];
    
    id<MovieFactoryMethodProtocol> item2 = [[AbstractFactory sharedFactory] createProduct:@protocol(MovieFactoryMethodProtocol)];
    
    id<TakeoutFactoryMethodProtocol> item3 = [[AbstractFactory sharedFactory] createProduct:@protocol(TakeoutFactoryMethodProtocol)];
    

    总结

    更具体的做法是,各个业务线会拆包开发,各自维护自己的业务线协议,新增业务线只需要在壳工程配置自己的协议即可,本文demo地址,该方案弥补了抽象工厂模式的拓展问题,但是存在字符串硬编码问题,若有类和协议不匹配会抛出异常,这里推荐阿里出品的iOS解耦神器BeeHive,它的做法是,用一个全局的字典保存协议与类名字的映射关系,当然这也是一种硬编码,只不过用起来更灵活,协议和类可以随意命名

    相关文章

      网友评论

        本文标题:抽象工厂模式-iOS实践

        本文链接:https://www.haomeiwen.com/subject/hvpwhrtx.html