`
bluepeer
  • 浏览: 74456 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

objc’s self and super 详解

 
阅读更多

在objc中的类实现中经常看到这两个关键字”self”和”super”,以以前oop语言的经验,拿c++为例,self相当于this,super相当于调用父类的方法,这么看起来是很容易理解的。

 

以下面的代码为例:

 

@interface Person:NSObject {
     NSString*  name;
}
- ( void ) setName:(NSString*) yourName;
@end
 
@interface PersonMe:Person {
     NSUInteger age;
}
- ( void ) setAge:(NSUInteger) age;
- ( void ) setName:(NSString*) yourName andAge:(NSUInteger) age;
@end
 
@implementation PersonMe
- ( void ) setName:(NSString*) yourName andAge:(NSUInteger) age {
     [self setAge:age];
     [super setName:yourName];
}
@end
 
int main( int argc, char * argv[]) {
     NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init]
     PersonMe* me = [[PersonMe alloc] init];
     [me setName:@ "asdf" andAge:18];
     [me release];
     [pool drain];
     return 0;
}

 

上面有简单的两个类,在子类PersonMe中调用了自己类中的setAge和父类中的setName,这些代码看起来很好理解,没什么问题。
然后我在setName:andAge的方法中加入两行:

 



NSLog(@ "self ' class is %@" , [self class ]);
NSLog(@ "super' class is %@" , [super class ]);

 

这样在调用时,会打出来这两个的class,先猜下吧,会打印出什么?
按照以前oop语言的经验,这里应该会输出:

self ' s class is PersonMe
super ' s class is Person

但是编译运行后,可以发现结果是:

self 's class is PersonMe
super ' s class is PersonMe

self的class和预想的一样,怎么super的class也是PersonMe?

真相

self 是类的隐藏的参数,指向当前当前调用方法的类,另一个隐藏参数是_cmd,代表当前类方法的selector。这里只关注这个self。super是个 啥?super并不是隐藏的参数,它只是一个“编译器指示符”,它和self指向的是相同的消息接收者,拿上面的代码为例,不论是用[self setName]还是[super setName],接收“setName”这个消息的接收者都是PersonMe* me这个对象。不同的是,super告诉编译器,当调用setName的方法时,要去调用父类的方法,而不是本类里的。

当使用self调用方法时,会从当前类的方法列表中开始找,如果没有,就从父类中再找;而当使用super时,则从父类的方法列表中开始找。然后调用父类的这个方法。

One more step

这种机制到底底层是如何实现的?其实当调用类方法的时候,编译器会将方法调用转成一个C函数方法调用,apple的objcRuntimeRef上说:

Sending Messages

When it encounters a method invocation, the compiler might generate a call to any of several functions to perform the actual message dispatch, depending on the receiver, the return value, and the arguments. You can use these functions to dynamically invoke methods from your own plain C code, or to use argument forms not permitted by NSObject’s perform… methods. These functions are declared in /usr/include/objc/objc-runtime.h.
■ objc_msgSend sends a message with a simple return value to an instance of a class.
■ objc_msgSend_stret sends a message with a data-structure return value to an instance of
a class.
■ objc_msgSendSuper sends a message with a simple return value to the superclass of an instance of a class.
■ objc_msgSendSuper_stret sends a message with a data-structure return value to the superclass of an instance of a class.

可以看到会转成调用上面4个方法中的一个,由于_stret系列的和没有_stret的那两个类似,先只关注objc_msgSend和objc_msgSendSuper两个方法。

当使用[self setName]调用时,会使用objc_msgSend的函数,先看下objc_msgSend的函数定义:

 



id objc_msgSend(id theReceiver, SEL theSelector, ...)


第一个参数是消息接收者,第二个参数是调用的具体类方法的selector,后面是selector方法的可变参数。我们先不管这个可变参数,以 [self setName:]为例,编译器会替换成调用objc_msgSend的函数调用,其中theReceiver是self,theSelector是 @selector(setName:),这个selector是从当前self的class的方法列表开始找的setName,当找到后把对应的 selector传递过去。

而当使用[super setName]调用时,会使用objc_msgSendSuper函数,看下objc_msgSendSuper的函数定义:

 



id objc_msgSendSuper( struct objc_super *super, SEL op, ...)


第一个参数是个objc_super的结构体,第二个参数还是类似上面的类方法的selector,先看下objc_super这个结构体是什么东西:

 


struct objc_super {
     id receiver;
     Class superClass;
};

 

可以看到这个结构体包含了两个成员,一个是receiver,这个类似上面objc_msgSend的第一个参数receiver,第二个成员是记 录写super这个类的父类是什么,拿上面的代码为例,当编译器遇到PersonMe里setName:andAge方法里的[super setName:]时,开始做这几个事:

1,构建objc_super的结构体,此时这个结构体的第一个成员变量receiver就是PersonMe* me,和self相同。而第二个成员变量superClass就是指类Person,因为PersonMe的超类就是这个Person。

2,调用objc_msgSendSuper的方法,将这个结构体和setName的sel传递过去。函数里面在做的事情类似这样:从 objc_super结构体指向的superClass的方法列表开始找setName的selector,找到后再以 objc_super->receiver去调用这个selector,可能也会使用objc_msgSend这个函数,不过此时的第一个参数 theReceiver就是objc_super->receiver,第二个参数是从objc_super->superClass中找到 的selector

里面的调用机制大体就是这样了,以上面的分析,回过头来看开始的代码,当输出[self class]和[super class]时,是个怎样的过程。
当使用[self class]时,这时的self是PersonMe,在使用objc_msgSend时,第一个参数是receiver也就是self,也是 PersonMe* me这个实例。第二个参数,要先找到class这个方法的selector,先从PersonMe这个类开始找,没有,然后到PersonMe的父类 Person中去找,也没有,再去Person的父类NSObject去找,一层一层向上找之后,在NSObject的类中发现这个class方法,而 NSObject的这个class方法,就是返回receiver的类别,所以这里输出PersonMe。

当使用[super class]时,这时要转换成objc_msgSendSuper的方法。先构造objc_super的结构体吧,第一个成员变量就是self, 第二个成员变量是Person,然后要找class这个selector,先去superClass也就是Person中去找,没有,然后去Person 的父类中去找,结果还是在NSObject中找到了。然后内部使用函数objc_msgSend(objc_super->receiver, @selector(class)) 去调用,此时已经和我们用[self class]调用时相同了,此时的receiver还是PersonMe* me,所以这里返回的也是PersonMe。

Furthor more

在类的方法列表寻找一个方法时,还牵涉到一个概念类对象的isa指针和objc的meta-class概念,这里就不再详细介绍。

转自: http://web2.0coder.com

分享到:
评论

相关推荐

    IOS self和super详解实现原理及区别

    在iOS的Objective-C编程中,`self`和`super`是两个非常重要的关键字,它们在对象方法调用中起着至关重要的作用。本文将详细解释`self`和`super`的实现原理及其区别。 首先,我们要理解`self`和`super`的基本概念。`...

    iOS百度地图简单使用详解

    [super viewDidLoad]; // 初始化地图 self.mapView = [[BMKMapView alloc] initWithFrame:self.view.frame]; self.mapView.delegate = self; // 设置地图类型为卫星地图 self.mapView.mapType = ...

    Object-C_Runtime

    ### Objective-C_Runtime 运行机制详解 #### 一、Objective-C 概览 Objective-C是一种结合了C语言和Smalltalk面向对象技术的编程语言。它最初由Brad Cox和Tom Love在1980年代初开发,并由Stepstone公司进一步发展...

    LocationManager.zip

    《LocationManager在OC中的应用与实现详解》 在iOS开发中,LocationManager是Apple的CoreLocation框架中的核心组件,主要用于获取设备的地理位置信息。本文将深入探讨LocationManager的使用、配置以及常见问题,...

    详解iOS开发中app的归档以及偏好设置的存储方式

    self = [super init]; if (self) { self.name = [decoder decodeObjectForKey:@"name"]; self.age = [decoder decodeIntegerForKey:@"age"]; self.height = [decoder decodeDoubleForKey:@"height"]; } ...

    iOS开发中UIPopoverController的使用详解

    [super viewDidLoad]; // 创建内容控制器 YYMenuViewController *menuVC = [[YYMenuViewController alloc] init]; // 创建PopoverController self.popover = [[UIPopoverController alloc] ...

    iOS 键盘躲避库 MLInputDodger.zip

    《iOS 键盘躲避库——MLInputDodger详解》 在iOS应用开发中,当用户在文本输入框中输入时,弹出的键盘经常会遮挡到下方的输入元素,这给用户体验带来了不便。为了解决这个问题,开发者们通常需要编写自定义代码来...

    一道值得深入思考的iOS面试题详解

    NSLog(@"My name is:%@",self.name); } @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; id cls = [Spark class]; void *obj = &cls; [(__bridge id)obj speak]; } ``` ...

    iOS中wkwebView内存泄漏与循环引用问题详解

    [self.webView.configuration.userContentController addScriptMessageHandler:self name:@"messageHandler"]; ``` 这里,`self`作为消息处理器被添加到`WKUserContentController`中,而`WKUserContentController`...

    iOS实现MJRefresh下拉刷新(上拉加载)使用详解

    [super prepare]; // 添加动画图片... } ``` 在`prepare`方法中,你需要为不同状态(如`MJRefreshStateIdle`、`MJRefreshStatePulling`和`MJRefreshStateRefreshing`)设置相应的图片数组。 除了上述基本场景...

    添加iAd代码

    **添加iAd代码详解** 在iOS应用开发中,为了增加收入和提高用户参与度,开发者经常会选择集成广告。苹果公司提供了自己的广告平台——iAd,它允许开发者在App内展示高质量、与应用内容相关的广告。本篇文章将详细...

    iOS OC - MVVM开发模式

    **iOS OC - MVVM开发模式详解** 在iOS应用开发中,Model-View-ViewModel(MVVM)是一种流行的设计模式,它旨在提高代码的可测试性、可维护性和可读性。MVVM起源于WPF(Windows Presentation Foundation)领域,后来...

    IOS 中两种单例模式的写法实例详解

    self = [super init]; if (self) { // 初始化操作 } return self; } @end ``` 在这个实现中,`defaultManager`是一个静态全局变量,保证了只有一个实例存在。当第一次调用`+defaultManager`时,如果`...

    在cocos2d里面如何使用物理引擎box2d

    ### 在cocos2d中使用物理引擎Box2D:弹球应用详解 #### 一、简介 在cocos2d框架中集成Box2D物理引擎是游戏开发中常见且重要的技术之一,尤其对于那些需要真实物理效果的游戏而言更是如此。Box2D是一个开源的2D物理...

    详解iOS集成GoogleMap(定位、搜索)

    [super viewDidLoad]; self.locationManager = [[CLLocationManager alloc] init]; self.locationManager.delegate = self; [self.locationManager requestAlwaysAuthorization]; self.locationManager....

    IOS点击按钮隐藏状态栏详解及实例代码

    [super viewDidLoad]; // ... self.hideStatus = [UIApplication sharedApplication].statusBarHidden; // ... } ``` 3. **添加按钮并监听点击事件**: 创建一个按钮,设置其位置、标题和点击事件。当用户...

    iOS中使用NSProgress类来创建UI进度条的方法详解

    [self.progress addObserver:self forKeyPath:@"fractionCompleted" options:NSKeyValueObservingOptionNew context:nil]; // 模拟每秒完成一个单位的任务 NSTimer *timer = [NSTimer ...

    详解iOS页面传值(顺传 逆传)

    [super viewDidLoad]; self.view.backgroundColor = [UIColor whiteColor]; self.navigationItem.title = self.contents; } ``` **逆传(反向传递)** 逆传是指从子视图控制器向父视图控制器传递数据,常用于...

Global site tag (gtag.js) - Google Analytics