一道题引发的self和super

这个是那道题目,让写出输出的结果:

刚看到这一道题目的时候我的第一反应就是输出Son     Father。但是输出的结果是Son Son。

下面是解析:

我首先建立了两个类,一个Father,一个Son

Father.h:

//
//  Father.h
//  SonFather
//
//  Created by zhanggui on 15/7/16.
//  Copyright (c) 2015年 zhanggui. All rights reserved.
//

#import <Foundation/Foundation.h>

@interface Father : NSObject
{
    NSString *name;
}
-(void)setName:(NSString *)youname;

@end
 

Father.m

//
//  Father.m
//  SonFather
//
//  Created by zhanggui on 15/7/16.
//  Copyright (c) 2015年 zhanggui. All rights reserved.
//

#import "Father.h"

@implementation Father

-(void)setName:(NSString *)youname
{
    name = youname;
}
@end
   

Son.h

//
//  Son.h
//  SonFather
//
//  Created by zhanggui on 15/7/16.
//  Copyright (c) 2015年 zhanggui. All rights reserved.
//

#import "Father.h"

@interface Son : Father
{
    NSUInteger age;
}
-(void)setAge:(NSUInteger)youage;
-(void)setName:(NSString *)youname andAge:(NSUInteger)youage;

@end
 

Son.m

//
//  Son.m
//  SonFather
//
//  Created by zhanggui on 15/7/16.
//  Copyright (c) 2015年 zhanggui. All rights reserved.
//

#import "Son.h"

@implementation Son

-(id)init
{
    self = [super init];
    if (self) {
        NSLog(@"%@",NSStringFromClass([self class]));
        NSLog(@"%@",NSStringFromClass([super class]));
    }
    return self;
}
-(void)setName:(NSString *)youname andAge:(NSUInteger)youage
{
    [self setAge:youage];
    [super setName:youname];
}
-(void)setAge:(NSUInteger)youage
{
    age = youage;
}
@end

  

然后我加入下面两个输出:

-(void)setName:(NSString *)youname andAge:(NSUInteger)youage
{
    [self setAge:youage];
    [super setName:youname];
    NSLog(@"class is %@",[self class]);
    NSLog(@"super class is %@",[super class]);
}

  

输出的结果是:

class is Son
super class is Son

运行中init那里面输出的也是

Son
Son

其实,self是类的隐藏的参数(另一个隐藏参数是_cmd,代表当前类方法的selector),指向当前调用方法的类.super并不是一个隐藏的参数,它只是一个编译器指示符,它和self指向的是相同的消息接收者。

就像[self class]    [super class]。接收class这个方法的接收者都是Son这个对象。不同的是,super告诉编译器,当使用super时,要去调用父类的方法,而不是本类里的方法。

当使用self调用方法时,会从当前类中方法列表中查找,如果没有,就从父类方法中查找;而当使用super时,则直接从父类方法中查找,然后调用父类的方法。

更深一步

这种机制是当调用类方法的时候,编译器会将方法调用转成一个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.

因此,当使用[self init],会调用objc_msgSend。我们看一下objc_msgSend定义:

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

第一个参数是消息接收者,第二个参数是调用具体类方法的selector,后面是selector方法的可变参数。当执行[self class]时,编译器会替换成调用objc_msgSend的函数调用,其中theReceive是self,therSelector是@selector(class),这个selector是从当前self的类方法中查找的,找到后吧对应selector传递过去。

而当执行[super class],会调用objc_msgSendSuper函数,下边看一下objc_msgSendSuper函数定义:

 struct objc_super { id receiver; Class superClass; };

可以看到该结构体包含两个成员,一个是receiver,第二个是记录写下super这个类的父类是什么。举个例子,当调用哪[super setName:]时,开始做以下几件事

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

调用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] 时,这时的 self 是 Son,在使用 objc_msgSend 时,第一个参数是 receiver 也就是 self,也是 Son* son1 这个实例。第二个参数,要先找到 class 这个方法的 selector,先从 PersonMe 这个类开始找,没有,然后到 PersonMe 的父类 Person 中去找,也没有,再去 Person 的父类 NSObject 去找,一层一层向上找之后,在 NSObject 的类中发现这个 class 方法,而 NSObject 的这个 class 方法,就是返回 receiver 的类别,所以这里输出 PersonMe。

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

参考来源:

http://www.spasvo.com/news/html/201317102404_2.html

http://blog.csdn.net/itianyi/article/details/8678452

时间: 2024-08-14 06:29:25

一道题引发的self和super的相关文章

一道题引发的惨案

昨天在一个cocos2d-x的群里,有群友发了一个问题求答案,当时自已也一下子没想到什么好的写法,这时候有个群友写了一个比较少见的代码方法,当时一看到这代码,我有一种如糊灌顶的感受,因为自已平时基本没用这种写法,今日在别的群,我就发了这道题让大家讨论一下,结果没想到,引发了各种争吵和讨论,有写得不好却自以为是的,有写得好却看不起别人的写法,有写得很好却十分低调的,十分精彩,也算是为群里热闹了一下气氛,QQ群也变成了社会市井的一个浓缩了,呵呵. 以下贴上问题和昨天那位群友写的代码,我觉得精彩的地方

C语言中,头文件和源文件的关系(转)

简单的说其实要理解C文件与头文件(即.h)有什么不同之处,首先需要弄明白编译器的工作过程,一般说来编译器会做以下几个过程: 1.预处理阶段 2.词法与语法分析阶段 3.编译阶段,首先编译成纯汇编语句,再将之汇编成跟CPU相关的二进制码,生成各个目标文件 (.obj文件)4.连接阶段,将各个目标文件中的各段代码进行绝对地址定位,生成跟特定平台相关的可执行文件,当然,最后还可以用objcopy生成纯二进制码,也就是去掉了文件格式信息.(生成.exe文件) 编译器在编译时是以C文件为单位进行的,也就是

大端小端系统_union_栈的增长方向

一道题引发的思考: 1.看一下之前写的union的特点,理解一下共享内存的概念 2.栈的增长方向是从高地址向低地址增长(数组比较特别,a[0]在低地址,a[n-1]在高地址)(堆由低地址到高地址存储) 3.小端系统 指 低字节存在低地址中,高字节存在高地址中:大端系统反之. 如上,联合体变量是4个字节,当A={"ab"};即str[0] ~ str[3]依次为: 'a'  'b'  '0'  '0',其中str[0]在低地址. 默认小端系统的情况下,A.l的低字节 对应 低地址,即00

C中的C文件与h文件辨析

简单的说其实要理解C文件与头文件(即.h)有什么不同之处,首先需要弄明白编译器的工作过程,一般说来编译器会做以下几个过程:       1.预处理阶段 2.词法与语法分析阶段 3.编译阶段,首先编译成纯汇编语句,再将之汇编成跟CPU相关的二进制码,生成各个目标文件 (.obj文件) 4.连接阶段,将各个目标文件中的各段代码进行绝对地址定位,生成跟特定平台相关的可执行文件,当然,最后还可以用objcopy生成纯二进制码,也就是去掉了文件格式信息.(生成.exe文件) 编译器在编译时是以C文件为单位

C程序中头文件相互包含精华(转载)

C程序中头文件相互包含精华(网摘小结) 收藏(转自:http://blog.csdn.net/lingyun3429/archive/2010/04/27/5535191.aspx).h中一般放的是同名.c文件中定义的变量.数组.函数的声明,需要让.c外部使用的声明. 1)h文件作用 1 方便开发:包含一些文件需要的共同的常量,结构,类型定义,函数,变量申明: 2 提供接口:对一个软件包来说可以提供一个给外界的接口(例如: stdio.h). 2)h文件里应该有什么 常量,结构,类型定义,函数,

C语言中.h和.c文件解析(转载)

转载:http://www.cnblogs.com/laojie4321/archive/2012/03/30/2425015.html   简单的说其实要理解C文件与头文件(即.h)有什么不同之处,首先需要弄明白编译器的工作过程,一般说来编译器会做以下几个过程:       1.预处理阶段 2.词法与语法分析阶段 3.编译阶段,首先编译成纯汇编语句,再将之汇编成跟CPU相关的二进制码,生成各个目标文件 (.obj文件) 4.连接阶段,将各个目标文件中的各段代码进行绝对地址定位,生成跟特定平台相

C语言中有关外部函数调用的问题

首先指出一点,我们通常所说的编译器并非仅指编译器,确切来说是编译工具链,里面包括了预编译器.编译器.汇编器和连接器. 对于外部函数实体(处于调用函数所在源文件之外的其他源文件中的函数),是在链接过程中,才会被寻找和添加进程序,一旦没有找到函数实体,就会报错,无法成功链接. 而外部函数的声明(一般声明在头文件中)只是令程序顺利通过编译而已,此时并不需要搜索到外部函数的实体. 当然,外部函数实体所在源文件也需要被编译为目标文件,至于链接时 如何找到该函数实体,这由链接器完成. 另外,头文件和对应源文

C语言怎么写头文件?

C语言中.h和.c文件解析(很精彩)   简单的说其实要理解C文件与头文件(即.h)有什么不同之处,首先需要弄明白编译器的工作过程,一般说来编译器会做以下几个过程: 1.预处理阶段 2.词法与语法分析阶段 3.编译阶段,首先编译成纯汇编语句,再将之汇编成跟CPU相关的二进制码,生成各个目标文件 (.obj文件) 4.连接阶段,将各个目标文件中的各段代码进行绝对地址定位,生成跟特定平台相关的可执行文件,当然,最后还可以用objcopy生成纯二进制码,也就是去掉了文件格式信息.(生成.exe文件)

转-C语言中.h和.c文件解析

C语言中.h和.c文件解析(很精彩)   简单的说其实要理解C文件与头文件(即.h)有什么不同之处,首先需要弄明白编译器的工作过程,一般说来编译器会做以下几个过程:       1.预处理阶段 2.词法与语法分析阶段 3.编译阶段,首先编译成纯汇编语句,再将之汇编成跟CPU相关的二进制码,生成各个目标文件 (.obj文件) 4.连接阶段,将各个目标文件中的各段代码进行绝对地址定位,生成跟特定平台相关的可执行文件,当然,最后还可以用objcopy生成纯二进制码,也就是去掉了文件格式信息.(生成.e