Runtime things (message mechanism)

Runtime things (message mechanism)

[[163324]]

1. About runtime

I have used runtime to solve the problem of changing the global font in a project before, so I once again felt the power of runtime black magic. Now I have the opportunity to share some of my understanding of runtime. Object calling method is a frequently used function in Objective-C, that is, message passing. Objective-C is a superset of C, so unlike C, Objective-C uses dynamic binding, that is, runtime. I won't say much about Objective-C's message passing and message mechanism. Today I will mainly talk about dynamic methods, that is, function calling.

2. Several related functions

The following figure summarizes in detail the order of each function call and the prerequisites for execution

Message passing function calls

1. When an object receives a message that it cannot interpret, it first calls the

  1. + (BOOL)resolveInstanceMethod:(SEL)sel

This method will be executed at runtime if the IML of SEL is not found. This function gives the class the opportunity to add functions using class_addMethod. According to the documentation, if the added function code is implemented, it returns YES, and if it is not implemented, it returns NO. For example, I created a new project. First, I executed the doSomething1 method in the ViewController class. The code is as follows

  1. //  
  2. // ViewController.m  
  3. // RuntimeTest1  
  4. //  
  5. // Created by HenryCheng on 15/12/24.  
  6. // Copyright ? (Copyright symbol) 2015 www.igancao.com All rights reserved.  
  7. //  
  8.   
  9. # import   "ViewController.h"  
  10.   
  11. @interface ViewController ()
  12.   
  13. @end  
  14.   
  15. @implementation ViewController
  16.   
  17. - ( void )viewDidLoad {
  18. [ super viewDidLoad];
  19. [self performSelector: @selector (doSomething)];
  20. }
  21.   
  22. - ( void )didReceiveMemoryWarning {
  23. [ super didReceiveMemoryWarning];
  24. // Dispose of any resources that can be recreated.  
  25. }
  26.   
  27. @end  

Operation Results

  1. ** 2015-12-24   10 : 35 : 37.726 RuntimeTest1[ 1877 : 337842 ]-[ViewController doSomething]: unrecognized selector sent to instance 0x7fe9f3736680 **
  2. ** 2015-12-24   10 : 35 : 37.729 RuntimeTest1[ 1877 : 337842 ] *** Terminating app due to uncaught exception 'NSInvalidArgumentException' , reason: '-[ViewController doSomething]: unrecognized selector sent to instance 0x7fe9f3736680' **
  3. ***** First throw call stack:**

As expected, the program crashes because the doSomething method is not found. Next, we implement the + (BOOL)resolveInstanceMethod:(SEL)sel method in it and determine if SEL is doSomething and output add method here

  1. //  
  2. // ViewController.m  
  3. // RuntimeTest1  
  4. //  
  5. // Created by HenryCheng on 15/12/24.  
  6. // Copyright ? (Copyright symbol) 2015 www.igancao.com All rights reserved.  
  7. //  
  8.   
  9. # import   "ViewController.h"  
  10.   
  11. @interface ViewController ()
  12.   
  13. @end  
  14.   
  15. @implementation ViewController
  16.   
  17. - ( void )viewDidLoad {
  18. [ super viewDidLoad];
  19. [self performSelector: @selector (doSomething)];
  20. }
  21.   
  22. + (BOOL)resolveInstanceMethod:(SEL)sel {
  23. if (sel == @selector (doSomething)) {
  24. NSLog(@ "add method here" );
  25. return YES;
  26. }
  27. return [ super resolveInstanceMethod:sel];
  28. }
  29.   
  30. - ( void )didReceiveMemoryWarning {
  31. [ super didReceiveMemoryWarning];
  32. // Dispose of any resources that can be recreated.  
  33. }
  34.   
  35. @end  

Continue running and see the log

  1. ** 2015-12-24   10 : 47 : 24.687 RuntimeTest1[ 2007 : 382077 ] add method here**
  2. ** 2015-12-24   10 : 47 : 24.687 RuntimeTest1[ 2007 : 382077 ]-[ViewController doSomething]: unrecognized selector sent to instance 0x7f9568c331f0 **
  3. ** 2015-12-24   10 : 47 : 24.690 RuntimeTest1[ 2007 : 382077 ] *** Terminating app due to uncaught exception 'NSInvalidArgumentException' , reason: '-[ViewController doSomething]: unrecognized selector sent to instance 0x7f9568c331f0' **
  4. ***** First throw call stack:**

You can see that the program still crashes, but we can see that add method here is output, which means that our + (BOOL)resolveInstanceMethod:(SEL)sel method has been executed and entered the judgment. Therefore, here, we can do something to make this method respond so that it will not crash in the ***- (void)doesNotRecognizeSelector:(SEL)aSelector method. Next, we continue to operate as follows

  1. //  
  2. // ViewController.m  
  3. // RuntimeTest1  
  4. //  
  5. // Created by HenryCheng on 15/12/24.  
  6. // Copyright ? (Copyright symbol) 2015 www.igancao.com All rights reserved.  
  7. //  
  8.   
  9. # import   "ViewController.h"  
  10.   
  11. # import [objc/runtime.h] (Due to recognition issues, the angle brackets are changed to square brackets here)
  12.   
  13. @interface ViewController ()
  14.   
  15. @end  
  16.   
  17. @implementation ViewController
  18.   
  19. - ( void )viewDidLoad {
  20. [ super viewDidLoad];
  21. [self performSelector: @selector (doSomething)];
  22. }
  23.   
  24. + (BOOL)resolveInstanceMethod:(SEL)sel {
  25. if (sel == @selector (doSomething)) {
  26. NSLog(@ "add method here" );
  27. class_addMethod([self class ], sel, (IMP)dynamicMethodIMP, "v@:" );
  28. return YES;
  29. }
  30. return [ super resolveInstanceMethod:sel];
  31. }
  32. void dynamicMethodIMP (id self, SEL _cmd) {
  33. NSLog(@ "doSomething SEL" );
  34. }
  35.   
  36. - ( void )didReceiveMemoryWarning {
  37. [ super didReceiveMemoryWarning];
  38. // Dispose of any resources that can be recreated.  
  39. }
  40.   
  41. @end  

Imported and executed the class_addMethod method in + (BOOL)resolveInstanceMethod:(SEL)sel, then defined a void dynamicMethodIMP (id self, SEL _cmd) function, ran the project, and looked at the log

  1. ** 2015-12-24   11 : 45 : 11.934 RuntimeTest1[ 2284 : 478571 ] add method here**
  2. ** 2015-12-24   11 : 45 : 11.934 RuntimeTest1[ 2284 : 478571 ] doSomething SEL**

At this time, we found that the program did not crash, and it also output doSomething SEL, which means that we have successfully added a method to our class through the runtime. The class_addMethod method is defined as follows

  • cls adds the method to this class, that is, the class to which the method is added

  • name method name, you can choose anything you want

  • imp is the function that implements this method

  • types is a string that defines the return value type and parameter type of the number. For example, "v@:", where v is void, the return type is empty, @ represents the parameter, here it refers to id (self), here: refers to the method SEL (_cmd), for example, I define a function

  1. int newMethod (id self, SEL _cmd, NSString *str) {
  2.   
  3. return   100 ;
  4. }

Then the method to add this function should be ass_addMethod([self class], @selector(newMethod), (IMP)newMethod, "i@:@");

2. If the method is not found or added in + (BOOL)resolveInstanceMethod:(SEL)sel

The message continues to be passed down to - (id)forwardingTargetForSelector:(SEL)aSelector to see if there is an object that can execute this method. Let's rebuild a project and create a new class called SecondViewController, which has a - (void)secondVCMethod method as follows

  1. //  
  2. // SecondViewController.m  
  3. // RuntimeTest2  
  4. //  
  5. // Created by HenryCheng on 15/12/24.  
  6. // Copyright ? (Copyright symbol) 2015 www.igancao.com All rights reserved.  
  7. //  
  8.   
  9. # import   "SecondViewController.h"  
  10.   
  11. @interface SecondViewController ()
  12.   
  13. @end  
  14.   
  15. @implementation SecondViewController
  16.   
  17. - ( void )viewDidLoad {
  18. [ super viewDidLoad];
  19. // Do any additional setup after loading the view.  
  20. }
  21.   
  22. - ( void )secondVCMethod {
  23. NSLog(@ "This is secondVC method !" );
  24. }
  25.   
  26. - ( void )didReceiveMemoryWarning {
  27. [ super didReceiveMemoryWarning];
  28. // Dispose of any resources that can be recreated.  
  29. }
  30.   
  31. /*
  32. #pragma mark - Navigation
  33.  
  34. // In a storyboard-based application, you will often want to do a little preparation before navigation
  35. - (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {
  36. // Get the new view controller using [segue destinationViewController].
  37. // Pass the selected object to the new view controller.
  38. }
  39. */  
  40.   
  41. @end  

The project structure should be like this

Project Catalog

Now I want to call the - (void)secondVCMethod method in ViewController. We know that ViewController and SecondViewController have no inheritance relationship. If we follow the normal steps, the program will definitely crash directly because it cannot find the - (void)secondVCMethod method in ViewController.

  1. //  
  2. // ViewController.m  
  3. // RuntimeTest2  
  4. //  
  5. // Created by HenryCheng on 15/12/24.  
  6. // Copyright ? (Copyright symbol) 2015 www.igancao.com All rights reserved.  
  7. //  
  8.   
  9. # import   "ViewController.h"  
  10. # import   @interface ViewController ()
  11.   
  12. @end  
  13.   
  14. @implementation ViewController
  15.   
  16. - ( void )viewDidLoad {
  17. [ super viewDidLoad];
  18. [self performSelector: @selector (secondVCMethod)];
  19. }
  20.   
  21. - ( void )didReceiveMemoryWarning {
  22. [ super didReceiveMemoryWarning];
  23. // Dispose of any resources that can be recreated.  
  24. }
  25.   
  26. @end  

Operation Results

  1. ** 2015-12-24   13 : 54 : 44.314 RuntimeTest2[ 3164 : 835814 ]-[ViewController secondVCMethod]: unrecognized selector sent to instance 0x7fc3a8535c10 **
  2. ** 2015-12-24   13 : 54 : 44.317 RuntimeTest2[ 3164 : 835814 ] *** Terminating app due to uncaught exception 'NSInvalidArgumentException' , reason: '-[ViewController secondVCMethod]: unrecognized selector sent to instance 0x7fc3a8535c10' **
  3. ***** First throw call stack:**

Now let's process this message as follows

  1. //  
  2. // ViewController.m  
  3. // RuntimeTest2  
  4. //  
  5. // Created by HenryCheng on 15/12/24.  
  6. // Copyright ? (Copyright symbol) 2015 www.igancao.com All rights reserved.  
  7. //  
  8.   
  9. # import   "ViewController.h"  
  10. # import   @interface ViewController ()
  11.   
  12. @end  
  13.   
  14. @implementation ViewController
  15.   
  16. - ( void )viewDidLoad {
  17. [ super viewDidLoad];
  18. [self performSelector: @selector (secondVCMethod)];
  19. }
  20.   
  21. - (id)forwardingTargetForSelector:(SEL)aSelector {
  22. Class class = NSClassFromString(@ "SecondViewController" );
  23. UIViewController *vc = class . new ;
  24. if (aSelector == NSSelectorFromString(@ "secondVCMethod" )) {
  25. NSLog(@ "secondVC do this!" );
  26. return vc;
  27. }
  28.       
  29. return nil;
  30. }
  31.   
  32. + (BOOL)resolveInstanceMethod:(SEL)sel {
  33.   
  34. return [ super resolveInstanceMethod:sel];
  35. }
  36.   
  37. - ( void )didReceiveMemoryWarning {
  38. [ super didReceiveMemoryWarning];
  39. // Dispose of any resources that can be recreated.  
  40. }
  41. @end  

Operation Results

  1. ** 2015-12-24   14 : 00 : 34.168 RuntimeTest2[ 3284 : 870957 ]secondVC do   this !**
  2. ** 2015-12-24   14 : 00 : 34.169 RuntimeTest2[ 3284 : 870957 ] This is secondVC method!**

We will find that - (void)secondVCMethod is executed and the program does not crash. The reason is that at this step

  1. - (id)forwardingTargetForSelector:(SEL)aSelector {
  2. Class class = NSClassFromString(@ "SecondViewController" );
  3. UIViewController *vc = class . new ;
  4. if (aSelector == NSSelectorFromString(@ "secondVCMethod" )) {
  5. NSLog(@ "secondVC do this!" );
  6. return vc;
  7. }
  8.       
  9. return nil;
  10. }

When the - (void)secondVCMethod method is not found, the message continues to be passed until - (id)forwardingTargetForSelector:(SEL)aSelector, and then I create a SecondViewController object in it, and if this method exists, I return the SecondViewController object. This function is the message forwarding. Here we successfully pass the message to SecondViewController and let it execute, so the method is executed. At the same time, it is equivalent to completing a multiple inheritance!

3. ***

Of course, there are several more functions, which are clearly expressed in the above picture. If you are interested, you can try it yourself to see what the order of message delivery is. The above-mentioned knowledge is just the tip of the iceberg of runtime. The power of runtime black magic is far more than this, such as method swizzling, which is still very useful in actual project combat. I will introduce it later when I have time.

refer to

  • Objective-C Runtime Reference

  • Objective-C Runtime Programming Guide

<<:  Ask the App Store Beginner's Guide Everything you need to know is here!

>>:  Apple vs. FBI: Not as simple as black and white

Recommend

Marketing promotion ineffective? These two things were not done well!

Before discussing marketing, there is a very impo...

KFC’s private domain traffic operation strategy!

Speaking of KFC , many people think it is the bes...

Lagou front-end high-paying training camp

Lagou front-end high-paying training camp resourc...

up to date! Reference data of 39 information flow platforms!!

The November data of major information flow platf...

How do small and medium-sized enterprises choose server leasing?

How do small and medium-sized enterprises choose ...

How does APP do data analysis?

Nowadays, data analysis is a must-talk about ever...

Easily find the storage location of WeChat received files

I believe many people have encountered this situa...

APP operation: How to improve user retention rate?

There is an important data indicator in the produ...

Complete knowledge about information flow advertising on iQiyi channel!

Information flow ads are ads located in the updat...

How to create a hit title with over 100,000 views? Just master these 5 points!

Let’s take you to re-examine the meaning of title...