In the previous article exploring the structure of classes, you learned about the isa pointer and superclass pointer moves. Relationships between objects, classes, metaclasses, and root metaclasses; Class structure; Bits data structure in class; Ro, RW, RWE, etc. This article makes some additions on the basis of class structure exploration!

1. Class loading

In the previous exploration, we encountered some knowledge points related to class loading. Here we simply make some additions and explanations. The details of class loading will be explained in a later section!

1. Why is firstSubclass nil

For example, in the case from the last article, LGTeacher inherits from LGPerson, and LGPerson inherits from NSObject, so LGTeacher is a subclass of LGPerson. But when I print the data structure of LGPerson class class_rw_t, I find that its subclass is empty. Why? See below:

I thought LGTeacher was inherited from LGPerson, so I should have exported LGTeacher! One tweak is to initialize LGTeacher before initializing LGPerson! See below:

Why is LGTeacher successfully exported in this? Since the object needs to be initialized, the class also needs to be initialized (loading the class)! But when do you initialize a class? Here’s the difference between a lazy-loaded class and a non-lazy-loaded class.

To put it simply:

  • Lazy loading class: Not implemented+load()Method class, will be inThe first message is sentWhen loading
  • Non-lazily loaded classes:+load()Method class, will be inThe main functionBefore,dyldClass initialization is performed during dynamic library loading

In the example above, when the LGTeacher object is initialized, the message is sent, so the class is loaded. So we print LGTeacher in the firstSubclass of class_rw_t of the LGPerson class.

2.cls->instanceSize

In fact, when the object is initialized, the memory space calculation has been involved in the class loading knowledge! See the code below:

inline size_t instanceSize(size_t extraBytes) const {
        if (fastpath(cache.hasFastInstanceSize(extraBytes))) {
            return cache.fastInstanceSize(extraBytes);
        }

        size_t size = alignedInstanceSize() + extraBytes;
        // CF requires all objects be at least 16 bytes.
        if (size < 16) size = 16;
        return size;
    }
Copy the code

If ignedInstancesize () is configured in the ignedcache, and alignedInstanceSize() is configured in the ignedcache, then alignedInstanceSize() is configured in the ignedcache, and alignedInstanceSize() is configured in the ignedcache.

Why is that?

  • If the class is lazily loaded, the object is initialized and sentAlloc messageThe class is initialized and the implementation method of the class is calledrealizeClassWithoutSwift
  • If the class is not lazily loaded, the object is initialized atThe main functionBefore, the implementation method of the class is also calledrealizeClassWithoutSwift

In realizeClassWithoutSwit, set the fastInstanceSize method:

// Set fastInstanceSize if it wasn't set already.
    cls->setInstanceSize(ro->instanceSize);
Copy the code

So, CLS ->instanceSize must be 16 bytes aligned!

Conclusion:

  • Lazy loading class: Not implemented+load()Method class, will be inThe first message is sentWhen loading
  • Non-lazily loaded classes:+load()Method class, will be inThe main functionBefore,dyldClass initialization is performed during dynamic library loading
  • Whether it’sLazy loadingorThe lazy loading, calls the implementation method of the classrealizeClassWithoutSwift

The detailed class loading process will be added in a later article.

Attribute, member variable, instance variable

See the following example, which is the same as the example in the previous article, to explore the structure of the class: a member variable subject, two properties name and hobby, and two methods -(void)sayNB and + (void)say666.

@interface LGPerson : NSObject{
    NSString *subject;
}
@property (nonatomic, copy) NSString *name;
@property (nonatomic, strong) NSString *hobby;

- (void)sayNB;
+ (void)say666;
@end

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        LGPerson *t1 = [[LGPerson alloc] init]; 
    }
    return 0;
}
Copy the code

1. LLDB exploration

The two properties name and hobby are obtained by calling the properties() method in class_rw_t. See below:

When retrieving the method list, there are four more object methods, namely the two get\set methods, as shown in the following figure:

The system automatically adds the GET \set method to the property. At the same time, when obtaining the list of member variables, there are two more member variables, namely, _name and _hobby. See below:

It can be concluded that:

  • attribute = The get/set methods + The underlined member variable;
  • The method at the bottom ismethod_tStructure of the form presented, includingMethod names,Type encoding,Method implementation, includingType encodinginThe nature of OC objects and ISAHas also been explained in the.

2. Explore CPP source code

Clang can be used to compile m files into CPP files, so we can learn more about the underlying implementation principles. See the definition of LGPerson in the CPP source below:

#ifndef _REWRITER_typedef_LGPerson #define _REWRITER_typedef_LGPerson typedef struct objc_object LGPerson; typedef struct {} _objc_exc_LGPerson; #endif extern "C" unsigned long OBJC_IVAR_$_LGPerson$_name; extern "C" unsigned long OBJC_IVAR_$_LGPerson$_hobby; extern "C" unsigned long OBJC_IVAR_$_LGPerson$__age; struct LGPerson_IMPL { struct NSObject_IMPL NSObject_IVARS; NSString *subject; NSString *_name; NSString *_hobby; NSString *__age; // Define the attribute as _age}; // @property (nonatomic, copy) NSString *name; // @property (nonatomic, copy) NSString *hobby; // @property (nonatomic, copy) NSString *_age; // - (void)sayNB; // + (void)say666; /* @end */ // @implementation LGPerson static NSString * _I_LGPerson_name(LGPerson * self, SEL _cmd) { return (*(NSString **)((char *)self + OBJC_IVAR_$_LGPerson$_name)); } extern "C" __declspec(dllimport) void objc_setProperty (id, SEL, long, id, bool, bool); static void _I_LGPerson_setName_(LGPerson * self, SEL _cmd, NSString *name) { objc_setProperty (self, _cmd, __OFFSETOFIVAR__(struct LGPerson, _name), (id)name, 0, 1); } static NSString * _I_LGPerson_hobby(LGPerson * self, SEL _cmd) { return (*(NSString **)((char *)self + OBJC_IVAR_$_LGPerson$_hobby)); } static void _I_LGPerson_setHobby_(LGPerson * self, SEL _cmd, NSString *hobby) { (*(NSString **)((char *)self + OBJC_IVAR_$_LGPerson$_hobby)) = hobby; } static NSString * _I_LGPerson__age(LGPerson * self, SEL _cmd) { return (*(NSString **)((char *)self + OBJC_IVAR_$_LGPerson$__age)); } static void _I_LGPerson_set_age_(LGPerson * self, SEL _cmd, NSString *_age) { objc_setProperty (self, _cmd, __OFFSETOFIVAR__(struct LGPerson, __age), (id)_age, 0, 1); } // @endCopy the code
  • The three attributes defined generate three member variables, which are:_name._hobby.__age;
  • If the defined property itself is underlined, an additional underline is added, such as:__age;
  • Properties are generated automaticallyThe get/set methodsMember variables are not generatedThe get/set methods;
  • objc_setPropertyIs automatically called when setting instance variables.Objc_setProperty method. This method can be understood asSet methodThe underlying adapter, through a unified package, implementationSet methodThe unified entrance. In this sectionThe nature of OC objects and ISAHas been explained in.

3. Differences between attributes, member variables, and instance variables

  • Properties = underlined member variable + setter + getter method

  • Instance variables: Special member variables (instantiations of classes) that are not basic data classes

3.Runtime interview questions

Update regularly!!

1.class_getInstanceMethod

The test cases are as follows:

LGPerson two methods //- (void)sayHello; //+ (void)sayHappy; Class void lgInstanceMethod_classToMetaclass(class pClass){const char *className = class_getName(pClass); Class metaClass = objc_getMetaClass(className); Method method1 = class_getInstanceMethod(pClass, @selector(sayHello)); Method method2 = class_getInstanceMethod(metaClass, @selector(sayHello)); Method method3 = class_getInstanceMethod(pClass, @selector(sayHappy)); Method method4 = class_getInstanceMethod(metaClass, @selector(sayHappy)); LGLog(@"%s - %p-%p-%p-%p",__func__,method1,method2,method3,method4); }Copy the code

Result analysis:

  1. ClassObject method sayHello? There are!
  2. Yes and no in metaclassesObject method sayHello? No!
  3. ClassClass methods sayHappy? No!
  4. Whether the metaclass hasClass methods sayHappy? There are!

Results verified:

Summary: The instance in class_getInstanceMethod cannot be mistaken for a method that gets the object of the class. An explicit concept here is that an object is an instance of a class, and a class is an instance of a metaclass, so there is no class method in the underlying wrapper. If it’s a class, it’s an object method to get the class, and if it’s a metaclass, it’s an object method to get the metaclass. Auto const methods = CLS ->data()->methods(); .

2.class_getClassMethod

Introduce the following test case:

LGPerson two methods //- (void)sayHello; //+ (void)sayHappy; Class void lgClassMethod_classToMetaclass(class pClass){const char *className = class_getName(pClass); Class metaClass = objc_getMetaClass(className); Method method1 = class_getClassMethod(pClass, @selector(sayHello)); Method method2 = class_getClassMethod(metaClass, @selector(sayHello)); Method method3 = class_getClassMethod(pClass, @selector(sayHappy)); Method method4 = class_getClassMethod(metaClass, @selector(sayHappy)); LGLog(@"%s-%p-%p-%p-%p",__func__,method1,method2,method3,method4); }Copy the code

Result analysis:

  1. class_getClassMethodIs to obtainClass method, that is,Object methods of the metaclassIs obtained from the metaclass, so-(void)sayHelloIt must not exist! somethod1andmethod2It must be empty.
  2. method3? Although the class is passed in, the source code finds the metaclass through the class to get the object methods. somethod3It exists!
  3. method4There’s nothing to say. There must be!

Results verified:

Summary: The source code implementation of class_getClassMethod is shown in the following code. The metaclass is found by the class passed in, and if the metaclass is passed in, the metaclass is directly obtained by the class. Getting Method calls class_getInstanceMethod, which is the Method that gets the object. An object is an instance of a class, and a class is an instance of a metaclass, so there is no such thing as a class method in the underlying wrapper.

Method class_getClassMethod(Class cls, SEL sel) { if (! cls || ! sel) return nil; And return class_getInstanceMethod (CLS - > getMeta (), sel); } Class getMeta() { if (isMetaClassMaybeUnrealized()) return (Class)this; else return this->ISA(); }Copy the code

3.class_getMethodImplementation

Class, introducing the following test case:

LGPerson two methods //- (void)sayHello; //+ (void)sayHappy; Class void lgIMP_classToMetaclass(class pClass){const char *className = class_getName(pClass); Class metaClass = objc_getMetaClass(className); IMP imp1 = class_getMethodImplementation(pClass, @selector(sayHello)); IMP imp2 = class_getMethodImplementation(metaClass, @selector(sayHello)); IMP imp3 = class_getMethodImplementation(pClass, @selector(sayHappy)); IMP imp4 = class_getMethodImplementation(metaClass, @selector(sayHappy)); NSLog(@"%s-%p-%p-%p-%p", __func__,imp1,imp2,imp3,imp4); }Copy the code

Result analysis:

  1. ClassObject method sayHelloMethod implementation? There are!
  2. Yes and no in metaclassesObject method sayHelloMethod implementation? No!
  3. ClassClass methods sayHappyMethod implementation? No!
  4. Whether the metaclass hasClass methods sayHappyMethod implementation? There are!

Results verified:

Summary: Surprisingly, imp2 and imp3 are not empty, and the method implementation address is the same. Why? After analyzing the source code of the class_getMethodImplementation implementation, when obtaining the method implementation of the class, a slow method lookup process will be entered. If no method implementation is found, a default method implementation will be returned, which is _objc_msgForward message forwarding. First of all, metaclasses have no object methods, and classes have no class methods, so they return the same method implementation, which is message forwarding.

IMP class_getMethodImplementation(Class cls, SEL sel) { IMP imp; if (! cls || ! sel) return nil; lockdebug_assert_no_locks_locked_except({ &loadMethodLock }); imp = lookUpImpOrNilTryCache(nil, sel, cls, LOOKUP_INITIALIZE | LOOKUP_RESOLVER); // Translate forwarding function to C-callable external version if (! imp) { return _objc_msgForward; } return imp; }Copy the code

Supplement:

  1. objc_msgSend

Objc_msgSend is the entry point to the Objective-C messaging system, which is implemented in assembly, and all objective-C messages are converted to be sent using objc_msgSend.

  1. _objc_msgForward

_objc_msgForward is the entry point for message forwarding, which is also implemented in assembly. When a method looks for loopUpImpOrForward, the IMP for the corresponding Selector method is not found, Resolution and dynamic method (resolveInstanceMethod/resolveClassMethod) fail will return _objc_msgForward, on behalf of the need for message forwarding. By pointing the method implementation to _objc_msgForward, you can make the method forward the message directly.

  1. _objc_msgForward_impcache

Tracing the source code found by the slow method, we find that the _objc_msgForward returned by class_getMethodImplementation is different from the _objc_msgForward_impcache assigned by default in loopUpImpOrForward. At the same time the message is forwarded. Why is it different?

A description is found in the objc_msg_arm64.s assembly. _objc_msgForward is an externally callable function that returns something like method_getImplementation(). The _objc_msgForward_impcache is the actual stored function pointer method cache. Provide different message forwarding for different methods or different usage scenarios?

4. IsKindOfClass, isMemberOfClass

Introduction case:

void lgKindofDemo(void){
    BOOL re1 = [(id)[NSObject class] isKindOfClass:[NSObject class]];       //
    BOOL re2 = [(id)[NSObject class] isMemberOfClass:[NSObject class]];     //
    BOOL re3 = [(id)[LGPerson class] isKindOfClass:[LGPerson class]];       //
    BOOL re4 = [(id)[LGPerson class] isMemberOfClass:[LGPerson class]];     //
    NSLog(@"\n re1 :%hhd\n re2 :%hhd\n re3 :%hhd\n re4 :%hhd\n",re1,re2,re3,re4);

    BOOL re5 = [(id)[NSObject alloc] isKindOfClass:[NSObject class]];       //
    BOOL re6 = [(id)[NSObject alloc] isMemberOfClass:[NSObject class]];     //
    BOOL re7 = [(id)[LGPerson alloc] isKindOfClass:[LGPerson class]];       //
    BOOL re8 = [(id)[LGPerson alloc] isMemberOfClass:[LGPerson class]];     //
    NSLog(@"\n re5 :%hhd\n re6 :%hhd\n re7 :%hhd\n re8 :%hhd\n",re5,re6,re7,re8);
}
Copy the code

Running result:

Case-by-case analysis:

1.+ (BOOL)isKindOfClass

Source code implementation is as follows:

+ (BOOL)isKindOfClass:(Class)cls {
    for (Class tcls = self->ISA(); tcls; tcls = tcls->getSuperclass()) {
        if (tcls == cls) return YES;
    }
    return NO;
}
Copy the code
  • [(id)[NSObject class] isKindOfClass:[NSObject class]];

    1. To obtainNSObjectClass, at this pointtclsIs the root metaclass,clsIs the root class, so it’s not equal;
    2. forCycle,tclsIs equal to the parent class of the root metaclass, which is the root class;
    3. The second judgment in the loop is returnedYES;
    4. The final result isYES.

    The path is: root class(self->ISA()) -> root metaclass (getSuperclass()) -> root class == root class [NSObject class]

  • [(id)[LGPerson class] isKindOfClass:[LGPerson class]];

    1. To obtainLGPersonClass, at this pointtclsisLGPersonMetaclasses,clsisLGPersonClass, so it’s not equal;
    2. forCycle,tclsEqual to the root metaclass, not equal;
    3. forCycle,tclsEqual to the root class, not equal;
    4. forCycle,tclsIs equal to thenilIs not equal;
    5. The final result isNO.

    The path is: LGPerson class(self->ISA()) -> LGPerson metaclass (getSuperclass()) -> root metaclass (getSuperclass()) -> root class(getSuperclass()) -> nil! = LGPerson class [LGPerson class]

2.+ (BOOL)isMemberOfClass

Source code implementation is as follows:

+ (BOOL)isMemberOfClass:(Class)cls {
    return self->ISA() == cls;
}
Copy the code
  • [(id)[NSObject class] isMemberOfClass:[NSObject class]];

    The root metaclass is compared to the root class, returning NO.

  • [(id)[LGPerson class] isMemberOfClass:[LGPerson class]];

    The LGPerson metaclass is compared to the LGPerson class and returns NO.

3.- (BOOL)isKindOfClass

Source code implementation is as follows:

- (BOOL)isKindOfClass:(Class)cls {
    for (Class tcls = [self class]; tcls; tcls = tcls->getSuperclass()) {
        if (tcls == cls) return YES;
    }
    return NO;
}

- (Class)class {
    return object_getClass(self);
}
Copy the code
  • [(id)[NSObject alloc] isKindOfClass:[NSObject class]];

    1. NSObjectObject gets the class, gets the root class, equals[NSObject class];
    2. Returns on the first time in the loopYES.
  • [(id)[LGPerson alloc] isKindOfClass:[LGPerson class]];

    1. LGPersonObject gets class, getsLGPersonClass that is equal to the[LGPerson class];
    2. Returns on the first time in the loopYES.
4.- (BOOL)isMemberOfClass

Source code implementation is as follows:

- (BOOL)isMemberOfClass:(Class)cls {
    return [self class] == cls;
}


- (Class)class {
    return object_getClass(self);
}
Copy the code
  • [(id)[NSObject alloc] isMemberOfClass:[NSObject class]];

    The root class object gets the class, and when compared to the root class, returns YES.

  • [(id)[LGPerson alloc] isMemberOfClass:[LGPerson class]];

    The LGPerson object fetching class is compared to the LGPerson class and returns YES.