使用工厂模式定制UITableViewCell

背景:最近在进行list页重构,list页的cell会有不同的展现形式,即同一个UITableView上多种cell并存。为了响应这种需求,我们考虑通过cell容器化的方式解决该问题。最理想的解决方法就是通过工厂模式来定制这些cell,当服务端告知我们某一个indexPath的cell的style时,我们就用相应类型的cell去填充。

 

工厂模式介绍

工厂模式可以分为简单工厂模式, 工厂方法模式, 抽象工厂模式,这三种模式在设计程度上由简单到复杂。下面挨个解释下各自的特点。

 

简单工厂模式

简单工厂模式是由工厂类直接生产相应的产品(通过类方法),然后在方法中通过switch-case 语句(或者if-else 语句)来决定胜场什么产品。该模式算是工厂模式的一个特例,因为当用户需要新增一种产品时,需要在直接修改工厂方法的switch-case 语句(或if-else 语句),通过添加分支条件来适应更多种情况,由此看出,它违背了开放-封闭原则。

@interface TRIPHotelCellFactory : NSObject

/**

 *  cell工厂方法

 *

 *  @param cellClassName 待新建的cell类名

 *  @param cellModel     cell数据模型

 *  @param indexPath     index索引

 *

 *  @return 目标cell

 */

+ (TRIPHotelBasicCell *)creatCellWithClassName:(NSString *)cellClassName

                                     cellModel:(TRIPHotelCellModel *)cellModel

                                     indexPath:(NSIndexPath *)indexPath;

@end

 

@implementation TRIPHotelCellFactory

 

+ (TRIPHotelBasicCell *)creatCellWithClassName:(NSString *)cellClassName

                                     cellModel:(TRIPHotelCellModel *)cellModel

                                     indexPath:(NSIndexPath *)indexPath{

    TRIPHotelBasicCell *cell = nil;

    

    if ([cellClassName isEqualToString:@"TRIPHotelStandardCell"]) {

        cell = [[TRIPHotelStandardCell alloc] initWithStyle:UITableViewCellStyleDefault

                                            reuseIdentifier:@"TRIPHotelStandardCell"];

    } else if ([cellClassName isEqualToString:@"TRIPHotelForSaleCell"]){

        cell = [[TRIPHotelForSaleCell alloc] initWithStyle:UITableViewCellStyleDefault

                                           reuseIdentifier:@"TRIPHotelForSaleCell"];

    }

    

    return [cell initData:cellModel indexPath:indexPath];

}

@end

 

工厂方法模式

工厂方法模式不直接生产产品,而是定义好如何生产产品的接口,然后由其子类决定具体生产何种产品,即工厂方法使一个类的实例化延迟到其子类。基类工厂只负责定义接口即可,具体的实现由其子类完成,当需要新增一种产品类型时,只需要再定义一个子类工厂,由该子工厂类去生产相应的产品。很好的满足了开发-封闭的原则。

@interface TRIPHotelCellFactory : NSObject

/**

 *  cell工厂方法

 *

 *  @param cellModel 待新建的cell类名

 *  @param indexPath cell数据模型

 *

 *  @return 目标cell

 */

+ (TRIPHotelBasicCell *)creatCellWithModel:(TRIPHotelCellModel *)cellModel

                                     indexPath:(NSIndexPath *)indexPath;

@end

 

@implementation TRIPHotelCellFactory

+ (TRIPHotelBasicCell *)creatCellWithModel:(TRIPHotelCellModel *)cellModel

                                 indexPath:(NSIndexPath *)indexPath{

    return [[TRIPHotelBasicCell alloc] initWithStyle:UITableViewCellStyleDefault

                                     reuseIdentifier:@"TRIPHotelBasicCell"];

}

@end

 

@interface TRIPHotelForSaleCellFactory : TRIPHotelCellFactory

 

@end

 

@implementation TRIPHotelForSaleCellFactory

 

+ (TRIPHotelBasicCell *)creatCellWithModel:(TRIPHotelCellModel *)cellModel

                                 indexPath:(NSIndexPath *)indexPath{

    TRIPHotelForSaleCell *cell = [[TRIPHotelForSaleCell alloc] initWithStyle:UITableViewCellStyleDefault

                                                             reuseIdentifier:@"TRIPHotelForSaleCell"];

    return [cell initData:cellModel indexPath:indexPath];

}

 

@end

实际上基类工厂方法中返回什么都无所谓,最终我们是其子类去生产相应的产品。

 

抽象工厂模式

该工厂模式就更复杂一些了,该工厂类不仅可以生产产品A,还可以生产产品B,只不过基类工程定义好生产相应产品的方法,尤其子类去实现具体的产品胜场过程。这个模式用的不是很多,这里就不举例了。

 

最佳实践

从上面的对比可以发现,简单工厂模式和工厂方法模式各有好处,一个实现简单,一个满足开放-封闭原则,支持封闭扩展,如果把这两者的好处融合在一起?

反射机制的引入是这种设想变成可能。

在oc中,反射形如:

NSString *cellClassName = @"TRIPHotelStandardCell";

Class classForCell = NSClassFromString(cellClassName);

可以直接通过类名即可获得相应的类并实现初始化。

 

修改简单工厂模式如下:

@implementation TRIPHotelCellFactory

 

+ (TRIPHotelBasicCell *)creatCellWithClassName:(NSString *)cellClassName

                                     cellModel:(TRIPHotelCellModel *)cellModel

                                     indexPath:(NSIndexPath *)indexPath{

    TRIPHotelBasicCell *cell = nil;

    

    // 通过反射来定义cell,当遇到cell拓展时,可以直接用字符串反射,无需修改该工厂方法

    Class classForCell = NSClassFromString(cellClassName);

    

    // 初始化目标cell

    cell = [[classForCell alloc] initWithStyle:UITableViewCellStyleDefault

                               reuseIdentifier:cellClassName];

    

    return [cell initData:cellModel indexPath:indexPath];

}

 

@end

这样一来,就可以在外部直接传入待生产cell的类名,即可获得相应的对象。当新增cell类型时,只需要在外部传入新定制的cell类名。该做法使简单工厂模式满足了封闭-开放原则,而且实现简单。

其实现形如:

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {

    TRIPHotelCellModel *dataModel = (TRIPHotelCellModel *)self.modelArray[indexPath.row];

    TRIPHotelBasicCell *cell = nil;

    switch (dataModel.cellStyle) {

        case HotelCellStyleStandard:

        {

            cell = [tableView dequeueReusableCellWithIdentifier:@"TRIPHotelStandardCell"];

            if (nil == cell) {

                cell = [TRIPHotelCellFactory creatCellWithClassName:@"TRIPHotelStandardCell"

                                                          cellModel:dataModel

                                                          indexPath:indexPath];

            }

        }

            break;

        case HotelCellStyleForSale:

        {

            cell = [tableView dequeueReusableCellWithIdentifier:@"TRIPHotelForSaleCell"];

            if (nil == cell) {

                cell = [TRIPHotelCellFactory creatCellWithClassName:@"TRIPHotelForSaleCell"

                                                          cellModel:dataModel

                                                          indexPath:indexPath];

            }

        }

            break;

        default:

            break;

    } 

    return cell;

}

 

posted on 2015-11-03 00:51  恒山之阳  阅读(1234)  评论(0编辑  收藏  举报

导航