ChatGPT解决这个技术问题 Extra ChatGPT

Static constructor equivalent in Objective-C?

I'm new to Objective C and I haven't been able to find out if there is the equivalent of a static constructor in the language, that is a static method in a class that will automatically be called before the first instance of such class is instantiated. Or do I need to call the Initialization code myself?

Thanks


Q
Quinn Taylor

The +initialize method is called automatically the first time a class is used, before any class methods are used or instances are created. You should never call +initialize yourself.

I also wanted to pass along a tidbit I learned that can bite you down the road: +initialize is inherited by subclasses, and is also called for each subclasses that doesn't implement an +initialize of their own. This can be especially problematic if you naively implement singleton initialization in +initialize. The solution is to check the type of the class variable like so:

+ (void) initialize {
  if (self == [MyParentClass class]) {
    // Once-only initializion
  }
  // Initialization for this class and any subclasses
}

All classes that descend from NSObject have both +class and -class methods that return the Class object. Since there is only one Class object for each class, we do want to test equality with the == operator. You can use this to filter what should happen only once ever, versus once for each distinct class in a hierarchy (which may not yet exist) below a given class.

On a tangential topic, it's worth learning about the following related methods, if you haven't already:

- isMemberOfClass:(Class)aClass (true only for aClass itself)

- isKindOfClass:(Class)aClass (true for aClass and children)

+ isSubclassOfClass:(Class)aClass (same as above, but a class method)

Edit: Check out this post by @bbum that explains more about +initialize: http://www.friday.com/bbum/2009/09/06/iniailize-can-be-executed-multiple-times-load-not-so-much/

Also, Mike Ash wrote a nice detailed Friday Q&A about the +initialize and +load methods: https://www.mikeash.com/pyblog/friday-qa-2009-05-22-objective-c-class-loading-and-initialization.html


"if ([self class] == [MyParentClass class])" [self class] is redundant here. you can just say if (self == [MyParentClass class])
Thank you! Your tidbit answered my question of why a particular static initializer was being invoked twice.
N
Nathan Kinsinger

There is the +initialize class method that will be called before a class is used.


Thank you, that's exactly what I was looking for, but I did searches to "static init", "static initializer", etc and didn't find it.
In almost every case, where in Java you'd say "static," you say "class" in Objective-C.
R
Richard J. Ross III

A bit of an addendum to this topic:

There is another way to create a 'static constructor' in obj-c, using an __attribute directive:

// prototype
void myStaticInitMethod(void);

__attribute__((constructor))
void myStaticInitMethod()
{
    // code here will be called as soon as the binary is loaded into memory
    // before any other code has a chance to call +initialize.
    // useful for a situation where you have a struct that must be 
    // initialized before any calls are made to the class, 
    // as they would be used as parameters to the constructors.
    // e.g.
    myStructDef.myVariable1 = "some C string";
    myStructDef.myFlag1 = TRUE; 

    // so when the user calls the code [MyClass createClassFromStruct:myStructDef], 
    // myStructDef is not junk values.
}

+load will do the same thing and looks more in sync with the Objective-C paradigm
@Grady Nope, I specifically list a situation in the post where +load is NOT equivalent.
@RichardJ.RossIII: according to the documentation for +load, +load methods are called before __attribute__(constructor) functions
@RichardJ.RossIII I don't understand what you are achieving since the code in the +load method is executed before the code in a function decorated with the constructor attribute. Can you please explain? Perhaps in an edit on your answer.
@Sam did you understand, what is RichardJ.Rosslll talking about ? I'd like to have a real-life example...