1

在 Objective-C 中,如何使用 ARC 编写单例?在ARC中,不允许覆盖release、autorelease、retain、retainCount方法,如何避免一个对象被释放?我知道没有 ARC,一个经典的单身人士会喜欢下面:

@interface SingletonObject
+ (SingletonObject*)sharedObject;
@end

SingletonObject *sharedObj;

@implementation SingletonObject
+ (id)allocWithZone:(NSZone *)zone
{
    if (sharedObj == nil) {
        //So the code [[SingletonObject alloc] init] is equal with [SingletonObject sharedObject]
        sharedObj = [super allocWithZone:zone];
    }
    return sharedObj; 
}


+ (void)initialize
{
    if (self == [SingletonObject class]) {
        sharedObj = [[SingletonObject alloc] init];
    }
}

+ (SingletonObject*)sharedObject
{
    return sharedObj;
}
- (id)retain
{
    return self;
}

- (unsigned)retainCount
{
    return UINT_MAX;  //denotes an object that cannot be released
}

- (oneway void)release
{
    //do nothing
}

- (id)autorelease
{
    return self;
}

- (id)init {
    self  = [super init];
    if (self) {
       //... 
    }
    return self;
}

@end

仅删除保留、保留计数、释放、自动释放方法是否安全?谢谢!

4

2 回答 2

8

你只需要一个方法来实现一个支持单例模式的类:

+ (instancetype)sharedInstance
{
    static id _sharedInstance = nil;
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        _sharedInstance = [[self alloc] init];
    });
    return _sharedInstance;
}

将其复制/粘贴到任何类中,它将有一个共享实例。除此之外的任何代码都只是增加了实际上没有必要的复杂性。我有时会补充:

- (void)dealloc
{
     *(char*)0x42 = 'b';
     // no super, ARC all the way
}

如果我的共享实例由于错误而被释放,那将导致非常具体的崩溃。(是的,十六进制 0x42 不是 42,但它在崩溃日志的寄存器中留下了一个不错的 0x000000042,从而可以立即识别发生了什么。)

于 2012-12-07T16:14:53.477 回答
0

是的。

-sharedObject不过应该是+sharedObject(类方法)。只要确保你总是使用它。

于 2012-12-07T03:41:04.510 回答