Singleton pattern in objc, how to keep init private?

后端 未结 4 1148
说谎
说谎 2020-12-19 23:47

How can i make sure user do not call init, instead client should call sharedSingleton to get a shared instance.

@synthesize delegate;

- (id)init
{
    self          


        
相关标签:
4条回答
  • 2020-12-19 23:54

    I've seen it done two ways.

    1. Throw an exception inside init.
    2. Have the object returned by init be your singleton object.

    Just to be clear, though, don't do this. It's unnecessary and will make your singletons overly difficult to test and subclass.

    edit to add examples

    Throw an exception in init

    - (instancetype)init {
        [self doesNotRecognizeSelector:_cmd];
        return nil;
    }
    
    - (instancetype)initPrivate {
        self = [super init];
        if (self) {
        }
        return self;
    }
    
    + (instancetype)sharedInstance {
        static MySingleton *sharedInstance;
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
            sharedInstance = [[self alloc] initPrivate];
        });
        return sharedInstance;
    }
    

    Have init return your singleton

    - (instancetype)init {
        return [[self class] sharedInstance];
    }
    
    - (instancetype)initPrivate {
        self = [super init];
        if (self) {
        }
        return self;
    }
    
    + (instancetype)sharedInstance {
        static MySingleton2 *sharedInstance;
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
            sharedInstance = [[self alloc] initPrivate];
        });
        return sharedInstance;
    }
    
    0 讨论(0)
  • 2020-12-20 00:05

    You can't make methods private in Objective-C. You could raise a NSException if the wrong initializer is invoked.

    - (id)init
    {
         [NSException exceptionWithName:@"InvalidOperation" reason:@"Cannot invoke init." userInfo:nil];
    }
    
    0 讨论(0)
  • 2020-12-20 00:15

    Short answer: you can't; Objective-C has no concept of private methods.

    Check out the answer to this similar question.

    0 讨论(0)
  • 2020-12-20 00:17

    Use UNAVAILABLE_ATTRIBUTE abolish init method, and implement initPrivate

    + (instancetype)shareInstance;
    
    - (instancetype)init UNAVAILABLE_ATTRIBUTE;
    + (instancetype)new UNAVAILABLE_ATTRIBUTE;
    

    implement

    + (instancetype)shareInstance {
        static MyClass *shareInstance = nil;
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
            shareInstance = [[super allocWithZone:NULL] initPrivate];
        });
        return shareInstance;
    }
    
    - (instancetype)initPrivate {
        self = [super init];
        if (self) {
    
        }
        return self;
    }
    
    //  MARK: Rewrite
    + (id)allocWithZone:(struct _NSZone *)zone {
        return [MyClass shareInstance];
    }
    
    - (id)copyWithZone:(NSZone *)zone
    {
        return self;
    }
    
    0 讨论(0)
提交回复
热议问题