Objective-C self super

题目

上题目,已知A是爷爷,B是爸爸,C是孙子。

@interface A : NSObject
- (void)f;
@end

@interface B : A
- (void)f;
- (void)g;
@end

@interface C : B
- (void)f;
@end

A,B,C都各自实现了函数f,只有B实现了函数g。

@implementation A
- (void)f
{
    NSLog(@"A");
}
@end

@implementation B
- (void)f
{
    NSLog(@"B");
}

- (void)g
{
    [self f];
    [super f];
    NSLog(@"%@", [self class]);
    NSLog(@"%@", [super class]);
}
@end

@implementation C
- (void)f
{
    NSLog(@"C");
}
@end

请问,下面代码输出什么?

C *c = [[C alloc] init];
[c g];

答案是CACC.

分析

1.C中没有g,为什么可以调用B的方法

C继承自B,所以C中虽然没有g方法,但是会去从下往上找,找打B类的g方法进行调用。

2.关于[self f]

调用B的g的过程中,self其实C,因为此时函数g消息的接受者是c。所以系统从self的类(即C)开始从下往上找,因为C中实现了f,所以直接输出C。

3.关于[super f]

这个问题是最困惑我的,卧槽,居然输出的是A而不是B?!

首先要明确的一点是,self和super不是同一个层面上的东西。

self是什么?

self 是类的隐藏的参数,指向当前当前调用方法的类,另一个隐藏参数是 _cmd,代表当前类方法的 selector。

super是什么?

super可不是想象中的self.superclass啊

super是Objective-C的语法糖,是一个编译器指示符,像宏一样,运行时XCode可不认识super这个东西。

 

其实Objective-C的函数调用的本质是

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

省略号表示不定参数。

即[self f]的本质是 objc_msgSend(self, f, "");

所以,这点跟C++不同,并不是self去调用了函数f,而是消息f发送给了self,self是消息接受者。正是因为这样,Objective-C中[nil f]这种写法没问题,因为发送给nil的任何消息都返回nil。

objc_msgSend(theReceiver, op, ...)的本质是,从theReceiver所在的类开始找,不断往上找,看看哪个类实现了op,一旦找到则将消息发送给它处理。

但是[super f] 可不就是这样了,[super f]本质是

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

省略号表示不定参数。

即[self f]的本质是 objc_msgSendSuper(super, f, "");

objc_super何许人也?

struct objc_super {
   __unsafe_unretained id receiver;
   __unsafe_unretained Class super_class;
};

objc_super的receiver是什么?

reveiver是当前消息的接受者,代码里写的是[c g],所以super的receiver就是C.你也可以理解为就是self。

objc_super的super_class是什么?

super_class是当前super所在代码的类的父类,很绕口,但是这句一定要好好理解。比如无论谁调用了g,是C也好是B也罢,g中[super f]的super的super_class一定是A,因为super这5个字母是写在B里面的。

objc_msgSendSuper(super, op, ...)的本质是,从super_class->super_class所在的类开始找,不断往上找,看看哪个类实现了op,一旦找到了则将消息发送给它处理。

在编译的时候,编译器看到了super,会因此构造一个struct。objc_super->receiver填了C,objc_super->super_class填了A。因此运行时编译器变不认识了super。

如果你已经清楚了上面的逻辑,请回头看看[super f]。

[super f],super->reveiver是C,super->super_class是A。所以从A开始找f,找到了则系统可能这样调用

objc_msgSend(super->recevier, f, nil)  //此时f是已经找到了的f,即A的f

4.关于[self class]

首先非常有必要看看class的实现,参考自苹果RunTime开源代码

- (Class)class {
    return object_getClass(self);
}

可以看到class返回的是self的类名。

好,继续,[self class]中,class也是一个消息,接受者是self,也就是C,所以系统从C开始找class函数的实现,没找到,找B,找A都找不到,到了NSObjcect这一层找到了。

此时调用了NSObject的class方法,那此时self是C,那就返回了C。

5.关于[super class]

这个问题一开始看起来也很奇怪,居然[self class]和[super class]输出了同一个东西,这不科学啊。但是把上面的知识理解好了,也就不奇怪了。

执行[super class]时,编译器将[super class]替换为

objc_msgSendSuper(super, class, nil)

其中super为reveiver=C,superclass=A。

系统从A开始找class,找不到,到了NSObject这一层找到了。

那么系统可能这样调用

objc_msgSend(objc_super->receiver, class, nil)  //此时class是已经找到了的class,即NSObject的class

此时调用了NSObject的class方法,那此时self是C,那就返回了C。

总结

道理懂了之后就是思考的效率问题了,凡是self的就从消息接受者的类开始从下往上找;凡是super的就从写super的这个类开始往上找。

参考链接

刨根问底Objective-C Runtime(1)- Self & Super

Objective-C 中Self 和 Super 详解

苹果RunTime开源代码

Objective-C Super的理解

时间: 2024-10-09 21:38:48

Objective-C self super的相关文章

关于 objective - C 的 super init

[super init] 的结果可能有三种: 第一种: [super init] 初始化成功,这个是最最正常的情况. 第二种: [super init] 初始化失败,我们都知道oc的两步创建,alloc开辟内存空间,init初始化对象,但是init还有另外一个作用,在初始化失败的时候,init方法会返回一个nil.回头行文的最上面,如果self为nil,那么if(self)语句的内容,将不被执行,已确保程序健壮安全. - (instancetype)init { self = [super in

刨根问底Objective-C Runtime(1)- Self & Super

刨根问底Objective-C Runtime(1)- Self & Super - Chun Tips Chun Tips 专注iOS开发 刨根问底Objective-C Runtime(1)- Self & Super 前言 关于Objective-C Runtime一篇好的文档 : Understanding the Objective-C Runtime 译文地址为: http://blog.cocoabit.com/blog/2014/10/06/yi-li-jieobjecti

四.OC基础--1.文档安装和方法重载,2.self和super&static,3.继承和派生,4.实例变量修饰符 ,5.私有变量&私有方法,6.description方法

四.OC基础--1.文档安装和方法重载, 1. 在线安装 xcode-> 系统偏好设置->DownLoads->Doucument->下载 2. 离线安装 百度xcode文档 3. 方法重载: 是指在一个类中定义多个同名的方法 在OC中没有重载 2.self和super&static, self和super: 1. self理解: 谁调用当前方法, self就代表谁. 比如: 在对象方法中,self代表的是对象, 因为只有对象才可以调用对象方法 在类方法中, self代表的

iOS完全自学手册——[三]Objective-C语言速成,利用Objective-C创建自己的对象

1.前言 上一篇已经介绍了App Delegate.View Controller的基本概念,除此之外,分别利用storyboard和纯代码创建了第一个Xcode的工程,并对不同方式搭建项目进行了比较.这一篇文章,我准备为大家介绍一下奇葩的Objective-C语言的语法和使用.这篇文章主要讲Objective-C的理论概念. 2.学习目标 2.1 Objective-C语言面向对象的特性与其他语言(C++)类比学习,声明定义一个MyObject类,并创建这个类的实例. 2.2 Objectiv

Objective C - 4 - 下载图片并且加载到View

? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 #import "LoadInternetImageViewController.h" @interface LoadInternetImageViewControll

iOS开发核心语言Objective C —— 全部知识点总结

本分享是面向有意向从事iOS开发的伙伴及苹果产品的发烧友,亦或是已经从事了iOS的开发人员,想进一步提升者.假设您对iOS开发有极高的兴趣,能够与我一起探讨iOS开发.一起学习,共同进步.假设您是零基础.建议您先翻阅我之前分享的iOS开发分分钟搞定C语言系列,然后在開始Objective C语言的学习,假设您遇到问题也能够与我探讨.另外将无偿分享自己整理出来的大概400G iOS学习视频及学习资料,都是干货哦.能够新浪微博私信?关注极客James,期待与您的共同学习和探讨! ! 由于时间有限,每

刨根问底Objective-C Runtime

http://chun.tips/blog/2014/11/05/bao-gen-wen-di-objective%5Bnil%5Dc-runtime-(2)%5Bnil%5D-object-and-class-and-meta-class/ 刨根问底Objective-C Runtime(1)- Self & Super 刨根问底Objective-C Runtime(2)- Object & Class & Meta Class 刨根问底Objective-C Runtime(

Objective -C Object initialization 对象初始化

Objective -C Object initialization 对象初始化 1.1 Allocating Objects  分配对象 Allocation is the process by which a new object is born. allocation 是新对象诞生的过程. Sending the alloc message to a class causes that class to allocate a chunk of memory large enough to

[super dealloc]内存释放的先后顺序

Objective-c 语言中最头疼的事就是内存释放,申明一个变量后记得一定要释放这个变量,相信很多人在dealloc函数[super dealloc]位置这问题上纠结过,经过实践发现,[super dealloc]写在自己释放的内存之前,经常会发生crash,而写在之后不会.对,Objective-c 中不能把自己写的释放内存放在[super dealloc]之后,原因是“你所创建的每个类都是从父类,根类继承来的,有很多实例变量也会继承过来,这部分变量有时候会在你的程序内使用,它们不会自动释放