Nature of Method Swizzling
Method Swizzling essentially swaps IMP and SEL. That’s what we call “dark magic.”
Method Swizzling principle
Method Swizzing happens at run time and is basically used to swap two methods at run time, we can write Method Swizzling code anywhere, but the swap only works after this Method Swilzzling code has run out.
And Method Swizzling is also an implementation of AOP(faceted programming) in __iOS__, and we can take advantage of apple’s feature to achieve AOP programming.
You can see it in two pictures
In Figure 1 above, selector2 originally corresponds to IMP2, but to make it easier to implement specific business requirements, we added selector3 and IMP3 to Figure 2, with selector2 pointing to IMP3 and selector3 pointing to IMP2, thus implementing the “method interchange”.
The runtime feature of the OC language calls an object by sending a message to the object. Is by finding the list of methods to receive the message object, from the list of methods to find the corresponding SEL, this SEL corresponds to an IMP(an IMP can correspond to multiple SEL), through the IMP to find the corresponding method call.
In each class there is a Dispatch Table, which essentially corresponds SEL and IMP(which can be understood as function pointer) in the class. Our Method Swizzling operates on this table and makes SEL correspond to another IMP.
Problems with using method-Swizzling
One-time problems in the use of method-Swizzling
The mehod-swizzling is written in the load method, and the load method will be called many times, which will cause the method to swap repeatedly, so that the method SEL pointing back to the original IMP problem
As shown in the figure:
In view of this situation, simple profit should be used to deal with
The subclass is not implemented, the superclass is implemented
LGPerson implements personInstanceMethod, while LGStudent inherits from LGPerson and does not implement personInstanceMethod. What’s wrong with running the following code?
Method interchange is implemented through the classification LG of LGStudent
Call the class
Running results:
- Cause analysis:
The crash point is [p personInstanceMethod]; The reason for this is that there’s a method swap in the LGStudent category, which swaps imp from Person to lg_studentIndtanceMethod from LGStudent, and then I need to go to LGPerson and find lg_studentInstanceMethod, However, LGPerson does not have this method, and the IMP cannot be found, causing a crash.
- To optimize the
The subclass is not implemented, and the superclass is not implemented
- To optimize the
Method Swizzling class cluster
During the development of our project, we often crashed because the NSArray array was out of bounds or the key or value value of NSDictionary was nil. For these problems, Apple would not give a warning, but directly crashed. It felt that Apple was really “too harsh”.
We can Swizzling NSArray, NSMutableArray, NSDictionary, NSMutableDictionary and other classes according to the above example. But… Method Swizzling doesn’t work and the code is correct. What the hell is that?
This is because Method Swizzling doesn’t work with NSArray and other classes. Because these classes, clusters of classes, are actually a kind of abstract factory design pattern. Inside the abstract factory there are many other subclasses that inherit from the current class, and the abstract factory class creates different abstract objects to use depending on the situation. For example, if we call NSArray’s objectAtIndex: method, this class will check inside the method and create different abstract classes to operate on.
So what we’re doing to the NSArray class is really just doing to the parent class, and inside NSArray we’re going to create other subclasses to do that, and it’s not really doing that to the NSArray itself, so we’re going to do that to the NSArray itself.
Code sample
Notice that __NSArrayI is the true class of NSArray, and NSMutableArray is different . We can get the real class from the Runtime function: objc_getClass(“__NSArrayI”);
Here are some examples of common class clusters:
class | “Mami” |
---|---|
NSArray | __NSArrayI |
NSMutableArray | __NSArrayM |
NSDictionary | __NSDictionaryI |
NSMutableDictionary | __NSDictionaryM |
The others can be searched
KVC
KVC concept
Key Value Coding, also known as KVC, is an important concept in iOS development, which translates to Key Value Coding in Chinese. The specific definition of this concept can be found in Apple’s official documents.
Key-value coding is a mechanism enabled by the NSKeyValueCoding informal protocol that objects adopt to provide indirect KVC is a mechanism enabled through an informal protocol, NSKeyValueCoding, which provides indirect access to its properties.Copy the code
Basic usage
- Directly evaluates and assigns values to attributes or member variables
- For collection properties, you can directly use mutableArrayValueForKey to make operational changes to an object’s collection properties
- KVC can also operate directly on structures, but the operation requires that the structure be converted to NSValue
- If the properties of the object are also objects, KVC can manipulate the properties of the object’s properties via keyPath
- If the key of the dictionary and the attribute of an object is the same, you can directly by setValuesForKeysWithDictionary will value assigned to the corresponding object properties of the dictionary, in the same way, But can be by dictionaryWithValuesForKeys transform objects into the dictionary
- You can use the API to get the length of an array element, or you can manipulate an array element to get a new array
- Aggregate operator
The underlying implementation of KVC
- KVC- The assignment process
According to the above official documents:
- Set < Key>: _set< Key>: setIs< Key>: call assignment if found.
- If there is no relevant methods, will see if class methods accessInstanceVariablesDirectly to YES to enter the next step. Otherwise, go to Step 4
- If the value is YES, you can directly access the member variables to perform the assignment, and find the variables _< key>, _is< key>, < key>, and is< key> in sequence. If found, directly assign, otherwise proceed to the next step.
- Will call * * setValue: forUndefinedKey: * * method throws an exception. The implementation can be customized for scenarios not found to avoid the behavior of throwing exceptions.
- KVC- The value process
ValueForKey: The method searches the object for patterns after the caller passes in the key as follows:
- In order to
get<Key>
.<key>
.is<Key>
As well as_<key>
To find if there are corresponding methods in the object.
- If found, attach the method return value and jump to step 5
- If not, go to step 2
- Find if there are
countOf<Key>
和objectIn<Key>AtIndex:
Method (corresponding toNSArray
The original method of the class definition) and<key>AtIndexes:
Method (corresponding toNSArray
methodsobjectsAtIndexes:
)
- If you find the first of these (
countOf<Key>
), and find at least one of the other two, create a response allNSArray
Method, and returns the object. (The translation is eithercountOf<Key>
+objectIn<Key>AtIndex:
, eithercountOf<Key>
+<key>AtIndexes:
, eithercountOf<Key>
+objectIn<Key>AtIndex:
+<key>AtIndexes:
) - If not, go to step 3
- Look for,
countOf<Key>
.enumeratorOf<Key>
和memberOf<Key>
These three methods (corresponding to the original method defined by the NSSet class)
- If all three methods are found, a response all is created
NSSet
Method, and returns the object - If not, go to Step 4
- Judgment class method
accessInstanceVariablesDirectly
The results of
- If the return
YES
, the_<key>
._is<Key>
.<key>
.is<Key>
Search for member variables in the order of, if found, add the member variable jump to step 5, if not, jump to step 6 - If the return
NO
, go to step 6
- Determine the value of the fetched attribute
- If the property value is an object, return it directly
- If the property value is not an object, but can be converted to
NSNumber
Type, the attribute value is converted toNSNumber
Type returns - If the property value is not an object, it cannot be converted to
NSNumber
Type, the attribute value is converted toNSValue
Type returns
- call
valueForUndefinedKey:
. By default, this throws an exception, butNSObject
Subclasses of can provide specific tokey
Behavior.
Conclusion:
KVC usage is shown below (IOS13 has some private property exceptions)