iOS创建Model的最佳实践

640?wx_fmt=gif

640?wx_fmt=jpeg

Linux编程 点击右侧关注,免费入门到精通! 640?wx_fmt=jpeg



作者丨大神Q
https://www.jianshu.com/p/7b9ac06f7ede


640?wx_fmt=gif

Immutable Model


我们以UserModle为例,我们可以像这样创建:


 
 

public class UserModelNSObject {

    public var userId: NSNumber
    public var name: String?
    public var email: String?
    public var age: Int?
    public var address: String?

    init(userId: NSNumber) {

        self.userId = userId

        super.init()
    }
}


用的时候可以像这样:


 
 

let userModel = UserModel(userId: 1)
user.email = "335050309@qq.com"
user.name = "roy"
user.age = 27
user.address = "上海市杨浦区"


这样创建一个User对象好处是弹性很大,我可以随意选择设定某个property的值,但是背后同样带有很大的缺点,就是这个Model变得异常开放,不安分,这种Model我们一般叫Mutable Model。有的时候我们需要Mutable Model,但大部分的时候出于数据安全和解耦考虑我们不希望创建的property在外部可以随意改变,在初始化后不可变的Model叫做Immutable Model,在开发中我的建议尽量使用Immutable Model。我们通过把property设置成readonly,在Swift可以用let或者private(set)。也就是这样:


 
 

public class UserModelNSObject {

    public let userId: NSNumber
    public private(setvar name: String?
    public private(setvar email: String?
    public private(setvar age: Int?
    public private(setvar address: String?

}


那么怎么写初始化方法呢?


Initializer mapping arguments to properties


当我们把property设置成readonly后,我们只能在init的时候赋值,这个时候就变成这样:


 
 

public class UserNSObject {

    public var userId: NSNumber
    public var name: String?
    public var email: String?
    public var age: Int?
    public var address: String?

    init(userId: NSNumber, name: String?, email: String, age: Int, address: String) {

        self.userId = userId

        super.init()

        self.name = name
        self.email = email
        self.age = age
        self.address = address
    }
}


使用的时候就变成这样:


 
 

let user = User.init(userId: 1, name: "335050309@qq.com", email: "roy", age: 27, address: "上海市杨浦区")


这样创建Model安全可靠,大多数时候是有效的,但是也有一些缺点:

如果property很多,init方法就有很多形参,然后变得又臭又长。

有的时候我们只需要Model的某些property,这样我们可能为各个不同的需求写不同的init方法,最终让UserModel变得很庞大。

Initializer taking dictionary

初始化的时候注入一个字典,就是下面的样子:


 
 

public class UserModelNSObject {

    public let userId: NSNumber
    public private(setvar name: String?
    public private(setvar email: String?
    public private(setvar age: Int?
    public private(setvar address: String?

    init(dic: NSDictionary) {

        self.userId = (dic["userId"as? NSNumber)!

        super.init()

        self.name = dic["name"as? String
        self.email = dic["email"as? String
        self.age = dic["age"asInt
        self.address = dic["address"as? String
    }
}


很显然这解决上一种第一个缺点,但是还是有一个不足之处:

如果字典没有某个属性对应的key的时候会崩溃,编译器并不能帮助我们排查这种运行时的崩溃。


不能很好的满足某些时候只需要Model的某些property的需求。


 640?wx_fmt=gif  Mutable subclass


我们看看Improving Immutable Object Initialization in Objective-C

https://holko.pl/2015/05/12/immutable-object-initialization/关于这个是怎么描述的


We end up unsatisfied and continue our quest for the best way to initialize immutable objects. Cocoa is a vast land, so we can – and should – steal some of the ideas used by Apple in its frameworks. We can create a mutable subclass of Reminder class which redefines all properties as readwrite:


 
 

@interface MutableReminder : Reminder <NSCopyingNSMutableCopying>

@property (nonatomiccopyreadwriteNSString *title;
@property (nonatomicstrongreadwriteNSDate *date;
@property (nonatomicassignreadwriteBOOL showsAlert;

@end


Apple uses this approach for example in NSParagraphStyle and NSMutableParagraphStyle. We move between mutable and immutable counterparts with -copy and -mutableCopy. The most common case matches our example: a base class is immutable and its subclass is mutable.


The main disadvantage of this way is that we end up with twice as many classes. What's more, mutable subclasses often exist only as a way to initialize and modify their immutable versions. Many bugs can be caused by using a mutable subclass by accident. For example, a mental burden shows in setting up properties. We have to always check if a mutable subclass exists, and if so use copy modifier instead of strong for the base class.


大致意思是创建一个可变子类,它将所有属性重新定义为readwrite。这种方式的主要缺点是我们最终得到两倍的类。而且,可变子类通常仅作为初始化和修改其不可变版本的方式存在。偶然使用可变子类可能会导致许多错误。例如,在设置属性时会出现心理负担。我们必须始终检查是否存在可变子类。


还有一点这种方式只能在Objective-C中使用。


640?wx_fmt=gifBuilder pattern


Builder pattern 模式需要我们使用一个Builder来创建目标对象,目标对象的property依旧是readonly,但是Builder的对应property却可以选择为readwrite。依旧用UserModel为例,我们需要为其进行适当的改造,改造之后:


 
 

typealias UserModelBuilderBlock = (UserModelBuilder) -> UserModelBuilder

public class UserModelNSObject{

    public let userId: NSNumber
    public private(setvar name: String?
    public private(setvar email: String?
    public private(setvar age: Int?
    public private(setvar address: String?

    init(userId: NSNumber) {

        self.userId = userId

        super.init()
    }

    convenience init(userId: NSNumber ,with block: UserModelBuilderBlock){

        let userModelBuilder = block(UserModelBuilder.init(userId: userId))
        self.init(userId: userModelBuilder.userId)
        self.email = userModelBuilder.email
        self.name = userModelBuilder.name
        self.age = userModelBuilder.age
        self.address = userModelBuilder.address
    }
}


之后是对应的Builder


 
 

class UserModelBuilderNSObject {

    public let userId: NSNumber
    public var name: String?
    public var email: String?
    public var age: Int?
    public var address: String?

    init(userId: NSNumber) {

        self.userId = userId
        super.init()
    }
}


然后可以像下面这样使用:


 
 

let userModle = UserModel(userId: 1) { (builder) -> UserModelBuilder in

    builder.email = "335050309@qq.com"
    builder.name = "roy"
    builder.age = 27
    builder.address = "上海市杨浦区"
    return builder
}


这种方式虽然我们需要为Model再创建一个Builder,略显啰嗦和复杂,但是当property较多,对Model的需求又比较复杂的时候这又确实是一种值得推荐的方式。


以上全是Swift的代码实现,下面我再贴上对应的OC代码


 
 

#import 

@interface RUserModelBuilder : NSObject

@property (nonatomicstrongreadwritenonnullNSNumber *userId;
@property (nonatomiccopyreadwritenullableNSString *name;
@property (nonatomiccopyreadwritenullableNSString *email;
@property (nonatomiccopyreadwritenullableNSNumber *age;
@property (nonatomiccopyreadwritenullableNSString *address;

@end

typedef RUserModelBuilder *__nonnull(^RUserModelBuilderBlock)(RUserModelBuilder *__nonnull userModelBuilder);

@interface RUserModel : NSObject

@property (nonatomicstrongreadonlynonnullNSNumber *userId;
@property (nonatomiccopyreadonlynullableNSString *name;
@property (nonatomiccopyreadonlynullableNSString *email;
@property (nonatomiccopyreadonlynullableNSNumber *age;
@property (nonatomiccopyreadonlynullableNSString *address;

+ (nonnull instancetype)buildWithBlock:(nonnull RUserModelBuilderBlock)builderBlock;

@end


 
 

#import "RUserModel.h"

@implementation RUserModelBuilder

@end

@interface RUserModel ()

@property (nonatomicstrongreadwritenonnullNSNumber *userId;
@property (nonatomiccopyreadwritenullableNSString *name;
@property (nonatomiccopyreadwritenullableNSString *email;
@property (nonatomiccopyreadwritenullableNSNumber *age;
@property (nonatomiccopyreadwritenullableNSString *address;

@end

@implementation RUserModel

#pragma mark - NSCopying

+ (nonnull instancetype)buildWithBlock:(nonnull RUserModelBuilderBlock)builderBlock {

    RUserModelBuilder *userModelBuilder = builderBlock([[RUserModelBuilder alloc] init]);

    RUserModel *userModel = [[RUserModel alloc] init];

    userModel.userId = userModelBuilder.userId;
    userModel.name = userModelBuilder.name;
    userModel.email = userModelBuilder.email;
    userModel.age = userModelBuilder.age;
    userModel.address = userModelBuilder.address;

    return userModel;
}

@end


demo地址

https://github.com/woshiccm/ImmutableModel/tree/master


 推荐↓↓↓ 

640?wx_fmt=png

?16个技术公众号】都在这里!

涵盖:程序员大咖、源码共读、程序员共读、数据结构与算法、黑客技术和网络安全、大数据科技、编程前端、Java、Python、Web编程开发、Android、iOS开发、Linux、数据库研发、幽默程序员等。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值