具有完全自定义 getter 和 setter 的属性的“弱”和“复制”属性
Posted
技术标签:
【中文标题】具有完全自定义 getter 和 setter 的属性的“弱”和“复制”属性【英文标题】:"weak" and "copy" attributes for property with full custom getter and setter 【发布时间】:2015-01-14 15:45:38 【问题描述】:我是 Objective-C 的初学者,所以我尝试通过这篇文章完成 Apple 练习:Apple Objective-C guide
我尝试使用自定义 getter 和 setter 创建类属性,一个具有“弱”属性,一个具有“复制”属性:
@property (weak, getter=getFirstName, setter=setFirstName:) NSString *firstName;
@property (copy, getter=getFirstName, setter=setFirstName:) NSString *secondName;
然后,我像这样测试这些属性:
NSString *name = @"John";
NSMutableString *surname = [NSMutableString stringWithString:@"Doe"];
MyPerson *person = [MyPerson createWithFirstName:name secondName:surname];
[person tellName];
name = nil;
[person tellName];
[NSThread sleepForTimeInterval:1.0f];
[surname appendString:@"dze"];
[person tellName];
所以我希望在 name 设置为 nil 之后,weak property 也变为 nil;并且标记为“副本”的属性将处理它们自己的初始字符串副本,并且将一些值附加到初始字符串不会在 MyPerson 对象中造成任何后果。但是,当我调用日志方法时,我有 firstName 的原始值和 secondName 中的更改值,就像没有“弱”和“复制”属性一样。
这是我的日志:
名字:约翰;第二名:Doe;出生日期:(空)
名字:约翰;第二名:Doe;出生日期:(空)
名字:约翰;第二名:Doedze;出生日期:(空)
我期待这样的事情:
名字:约翰;第二名:Doe;出生日期:(空)
名字:未指定;第二名:Doe;出生日期:(空)
名字:未指定;第二名:Doe;出生日期:(空)
我知道在第二行 GC 不能破坏 firstName 的值,但是在 1 秒睡眠后执行对 log 方法的第三次调用。我认为 1 秒足够 GC 收集未使用的字符串。
知道为什么可以忽略“弱”和“复制”属性吗? 感谢您的帮助。
完整代码: main.m:
#import <Foundation/Foundation.h>
#import "MyPerson.h"
int main(int argc, const char * argv[])
@autoreleasepool
NSString *name = @"John";
NSMutableString *surname = [NSMutableString stringWithString:@"Doe"];
MyPerson *person = [MyPerson createWithFirstName:name secondName:surname];
[person tellName];
name = nil;
[person tellName];
[NSThread sleepForTimeInterval:1.0f];
[surname appendString:@"dze"];
[person tellName];
return 0;
MyPerson.h:
#import <Foundation/Foundation.h>
@interface MyPerson : NSObject
@property (weak, getter=getFirstName, setter=setFirstName:) NSString *firstName;
@property (copy, getter=getSecondName, setter=setSecondName:) NSString *secondName;
@property (getter=getBirthDate, setter=setBirthDate:) NSDate *birthDate;
-(void)setFirstName:(NSString *)firstName;
-(NSString *)getFirstName;
-(void)setSecondName:(NSString *)secondName;
-(NSString *)getSecondName;
-(void)setBirthDate:(NSDate *)birthDate;
-(NSDate *)getBirthDate;
-(void) tellName;
+(id)create;
+(id)createWithFirstName:(NSString *)firstName;
+(id)createWithFirstName:(NSString *)firstName secondName:(NSString *)secondName;
+(id)createWithFirstName:(NSString *)firstName secondName:(NSString *)secondName birthDate:(NSDate *)birthDate;
-(id)initWithFirstName:(NSString *)firstName secondName:(NSString *)secondName birthDate:(NSDate *)birthDate;
@end
MyPerson.m:
#import "MyPerson.h"
@implementation MyPerson
@synthesize firstName = m_firstName;
@synthesize secondName = m_secondName;
@synthesize birthDate = m_birthDate;
-(void)tellName
NSMutableString *stringBuilder = [NSMutableString string];
[stringBuilder appendString:@"First name: "];
[self appendIfNotNil:self.firstName toBuilder:stringBuilder];
[stringBuilder appendString:@"; "];
[stringBuilder appendString:@"Second name: "];
[self appendIfNotNil:self.secondName toBuilder:stringBuilder];
[stringBuilder appendString:@"; "];
[stringBuilder appendString:@"BirthDate: "];
[stringBuilder appendFormat:@"%@", [m_birthDate descriptionWithLocale:NSLocaleLanguageDirectionUnknown]];
NSLog(@"%@", stringBuilder);
-(void)appendIfNotNil:(NSString*)str toBuilder:(NSMutableString *)stringBuilder
[stringBuilder appendString:str == nil ? @"Not specified" : str];
-(void)setFirstName:(NSString *)firstName
NSLog(@"Setter called for first name: %@", firstName);
m_firstName = firstName;
-(NSString *)getFirstName
NSLog(@"Getter called for first name");
return m_firstName;
-(void)setSecondName:(NSString *)secondName
NSLog(@"Setter called for second name: %@", secondName);
m_secondName = secondName;
-(NSString *)getSecondName
NSLog(@"Getter called for second name");
return m_secondName;
-(id)initWithFirstName:(NSString *)firstName secondName:(NSString *)secondName birthDate:(NSDate *)birthDate
self = [super init];
if (self)
m_firstName = firstName;
m_secondName = secondName;
m_birthDate = birthDate;
return self;
+(id)createWithFirstName:(NSString *)firstName secondName:(NSString *)secondName birthDate:(NSDate *)birthDate
MyPerson *person = [MyPerson alloc];
person = [person initWithFirstName:firstName secondName:secondName birthDate:birthDate];
return person;
+(id)createWithFirstName:(NSString *)firstName secondName:(NSString *)secondName
return [MyPerson createWithFirstName:firstName secondName:secondName birthDate:nil];
+(id)createWithFirstName:(NSString *)firstName
return [MyPerson createWithFirstName:firstName secondName:nil birthDate:nil];
+(id)create
return [MyPerson createWithFirstName:nil secondName:nil birthDate:nil];
@end
任何想法为什么可以忽略“弱”和“复制”属性?
【问题讨论】:
【参考方案1】:知道为什么可以忽略“弱”和“复制”属性吗?感谢您的帮助。
它们不会被忽略。你的测试有缺陷。
要测试“弱”,你需要使用 NSString 以外的东西,因为字符串有特殊的内存管理。使用没有变量引用的 NSObject;它会在一行之后消失在烟雾中。
self.ob = [NSObject new];
NSLog(@"%@", self.ob); // nil!
要测试“复制”,您需要从 NSMutableString 开始,保留对它的引用,将其分配给属性,然后改变可变字符串;分配的字符串不会改变,证明该属性没有持有对同一可变字符串的另一个引用。
要测试内存管理属性(如copy
)和线程属性(如atomic
),不要编写你自己的setter!这些是编译器关于 it 应该如何编写 setter(综合)的说明。如果您手动编写 setter,则不会进行综合,此时您的属性将毫无意义(除非它们通知客户端您的 API)。
【讨论】:
我书中的讨论可能会对你有所帮助:apeth.com/iosBook/… 这里是测试weak
的示例代码——注意我没有使用NSString! github.com/mattneub/Programming-iOS-Book-Examples/blob/master/…
matt,所以如果我在自定义设置器中使用“复制”,我必须在设置器中手动调用[参数复制]?这使得“复制”变得非常不必要。
是的,如果您编写一个副本自定义设置器,则由您来编写自定义设置器。您的@property
属性设置仅适用于synthesized setter。
如果我的回答未能解决这个问题,我很抱歉;我会为此添加注释。以上是关于具有完全自定义 getter 和 setter 的属性的“弱”和“复制”属性的主要内容,如果未能解决你的问题,请参考以下文章
如何在android中为自定义视图创建setter和getter
如何使用静态变量和自定义 getter 和 setter 在 SWIG 中扩展结构?
Kotlin自定义的getter和mutable list。
Intellij IDEA 自定义 getter and setter