gpt4 book ai didi

ios - 如何正确实现 mutableCopyWithZone 和 copyWithZone

转载 作者:可可西里 更新时间:2023-11-01 03:29:37 26 4
gpt4 key购买 nike

我读了其他一些关于它的主题,但我仍然迷路了。

我想创建两种对象,一种是只有“只读”属性的不可变对象(immutable对象),另一种是只有“读写”属性的可变对象。

我们称它们为 EXCar 和 EXMutableCar。

EXCar是NSObject的子类,EXMutableCar是EXCar的子类。

ExCar 的界面中会有

@property (nonatomic, strong, readonly) NSString *name;

EXMutableCar 的界面中会有

@property (nonatomic, strong) NSString *name;

所以当我使用它的子类 EXMutableCar 时,我“打开”了 EXCar 的属性。然后它是可变的。问题是在它们之间正确复制。

我在 EXCar 中实现了 mutableCopyWithZone :

- (id)mutableCopyWithZone:(NSZone *)zone {
EXMutableCar *mutableCopy = [[EXMutableCar allocWithZone:zone] init];
mutableCopy.name = _name;

return mutableCopy;
}

第一个问题,这样做的好方法吗? (我要吞文)

问题出在 copyWithZone 上。由于 EXCar 的属性是只读的,我不能在 EXCar 中创建,也不能在 EXMutableCar 中创建 EXCar 的新实例并像这样填充它的属性:

- (id)copyWithZone:(NSZone *)zone {
EXCar *copy = [[EXCar allocWithZone:zone] init];
copy.name = _name; // This can't work...

return copy;
}

而且我真的不想做一个有 15 个属性的“init”方法来传入(当然,EXCar 就是一个例子,真实的类充满了很多属性)。通常它们是从服务器的 JSON 消息启动的,因此它们不需要复杂的初始化方法。

第二个问题是,如何做一个让我的类不可变的 copyWithZone?

感谢您的帮助:)

最佳答案

代码:

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

@interface EXCar : NSObject <NSCopying, NSMutableCopying>

@property (nonatomic, strong, readonly) NSString* name;

@end

// EXCar.m
#import "EXCar.h"
#import "EXMutableCar.h"

@implementation EXCar

- (id)copyWithZone:(NSZone *)zone {
EXCar* car = [[[self class] allocWithZone:zone] init];
car->_name = [_name copyWithZone:zone];
return car;
}

- (id)mutableCopyWithZone:(NSZone *)zone {
EXMutableCar* mutableCar = [[EXMutableCar allocWithZone:zone] init];
mutableCar.name = [_name mutableCopyWithZone:zone];
return mutableCar;
}

@end

// EXMutableCar.h
#import "EXCar.h"

@interface EXMutableCar : EXCar

@property (nonatomic, strong) NSString* name;

@end

// EXMutableCar.m
#import "EXMutableCar.h"

@implementation EXMutableCar

@synthesize name = _mutableName;

- (id)copyWithZone:(NSZone *)zone {
EXMutableCar* car = [super copyWithZone:zone];
car->_mutableName = [_mutableName copyWithZone:zone];
return car;
}

- (id)mutableCopyWithZone:(NSZone *)zone {
EXMutableCar* car = [super mutableCopyWithZone:zone];
car->_mutableName = [_mutableName mutableCopyWithZone:zone];
return car;
}

说明:

  • EXCar接口(interface)实现了两种“复制”协议(protocol);
  • 子类EXMutableCar覆盖相同的属性,使其成为 readwrite .

EXMutableCar 中的第一件事实现:手动 @synthesize name因为 Xcode 给了我们一个警告,因为我们在父类(super class)中有相同的属性(但具有不同的访问说明符)。

请注意,我们可以为实例变量指定相同的名称,例如 _name , 但重要的是要理解我们在子类中声明了一个不同的变量,因为_name我们无法访问父类(super class)。

接下来,Apple 文档指出:

If a subclass inherits NSCopying from its superclass and declares additional instance variables, the subclass has to override copyWithZone: to properly handle its own instance variables, invoking the superclass’s implementation first.

NSMutableCopying 相同:

If a subclass inherits NSMutableCopying from its superclass and declares additional instance variables, the subclass has to override mutableCopyWithZone: to properly handle its own instance variables, invoking the superclass’s implementation first.

我们确实声明了额外的实例变量,所以我们也在我们的子类中覆盖了这些方法。

结果:

EXCar* car = [[EXCar alloc]init]; // car.name is (null)
EXCar* carCopy = [car copy]; // we can do this
EXMutableCar* mutableCar = [car mutableCopy]; // and this
mutableCar.name = @"BMW";
car = [mutableCar copy]; // car.name is now @"BMW"
EXMutableCar* anotherMutableCar = [car mutableCopy]; //anotherMutableCar.name is @"BMW"

关于ios - 如何正确实现 mutableCopyWithZone 和 copyWithZone,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18192220/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com