码迷,mamicode.com
首页 > 其他好文 > 详细

self和super之间的区别

时间:2015-12-19 21:46:49      阅读:188      评论:0      收藏:0      [点我收藏+]

标签:

 

关于 self 和 super 之间的区别, 

首先要了解  1, self  是什么 ;super 是什么。2,[ super init] 做什么。3,为什么要 self =  [super init];

1,self  是什么 ,super 是什么

   > 在动态方法中,self代表着"对象"

   > 在静态方法中,self代表着"类"

   > 万变不离其宗,记住一句话就行了:self代表着当前方法的调用者

self 和 super 是oc 提供的 两个保留字。 但有根本区别,

    self是类的隐藏的参数变量,指向当前调用方法的对象(类也是对象,类对象),另一个隐藏参数是_cmd,代表当前类方法的selector。

    super并不是隐藏的参数,它只是一个"编译器指示符"

2, [ super init]     做什么

发送消息时

Class  A
-reposition { ... [self setOrigin:someX :someY]; ... }
A  a= [a .. init];
[a  reposition];   方法体中  编译器将
  [self setOrigin:someX :someY];  

 其转换为

objc_msgSend(id self,SEL _cmd, ...) 。self -> a

 此时 self 指代a 对象,方法从a 对应 类结构的 方法调度表中开始寻找,如果找不到,延继承链往 父类中寻找 。

同样如果 reposition 是类方法, self 指代 A 类对象。

Class  A
-reposition  
{  
     ...  
     [super setOrigin:someX :someY];  
     ...  
}
[a  reposition];   方法体中编译器将
[super setOrigin:someX :someY]; 

其转换为

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 这个类的父类是什么,拿上面的代码为例,当编译器遇到 A 里

[super setOrigin:someX :someY]

 时,开始做这几个事:

      >构建 objc_super 的结构体,此时这个结构体的第一个成员变量 receiver 就是 a,和 self 相同。而第二个成员变量 superClass 就是指类 A的 superClass。

      >调用 objc_msgSendSuper 的方法,将这个结构体和

setOrigin

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

3,为什么要 self =  [super init];

符合oc 继承类 初始化规范 super 同样也是这样,  [super init]  去self 的super 中调用init     super 调用 superSuper 的init 。直到根类 NSObject 中的init ,

根类中init 负责初始化 内存区域  向里面添加 一些必要的属性,返回内存指针,  这样 延着继承链 初始化的内存指针 被从上 到 下 传递,在不同的子类中向块内存添加 子类必要的属性,直到 我们的 A 类中 得到内存指针,赋值给slef 参数, 在if (slef){//添加A 的属性 }

下面来看看这个:

@implementation Son : Father
- (id)init
{
    self = [super init];
    if (self)
    {
        NSLog(@"%@", NSStringFromClass([self class]));
        NSLog(@"%@", NSStringFromClass([super class]));
    }
    return self;
}
@end

应该不难分析出  打印结果:

Son
Son

当 发送 class 消息 时不管是 self 还是 super 其消息主体依然是 self ,也就是说 self 和 super 指向的 是同一个对象。只是 查找方法的位置 区别,一个从本类,一个从本类的超类。
一般情况下 class 方法 只有在 根类 NSObject 中定义,极少情况有子类重写 class 方法,
所以 [slef class] 和 [super class] 都是在 根类中 找方法实现, 消息接收主体 又都是 a
如果重写可能会不一样。
自然都打印出 Son

在来一个例子:
#import <Foundation/Foundation.h>

@interface EngineSuper : NSObject
-(void)printCurrentClass;

@end

#import "EngineSuper.h"

@implementation EngineSuper
-(void)printCurrentClass{

    NSLog(@"=EngineSuper=======%@",[self class]);
}
@end



@interface Engine : EngineSuper
-(void)printSupClass;
@end


@implementation Engine

-(void)printSupClass{
   [super printCurrentClass];
}

//调用:
Engine *engine = [[Engine alloc]init];
[engine  printCurrentClass];//直接调用父类 方法,engine没重载 它

[engine  printSupClass];//间接调用父类方法,

 

打印当然都是  :

Engine
Engine

方法体中 self 始终指代 方法的接收者 及对象 engine。,
 换成   NSLog(@"=EngineSuper=======%@",[super class]); 结果也是一样的。

 

首先主要的几点:

  1. self调用自己方法,super调用父类方法
  2. self是类,super是预编译指令
  3. 【self class】和【super class】输出是一样的

 

self和super底层实现原理:

  1. 当使用 self 调用方法时,会从当前类的方法列表中开始找,如果没有,就从父类中再找;而当使用 super 时,则从父类的方法列表中开始找,然后调用父类的这个方法。
  2. 当使用 self 调用时,会使用 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 传递过去。
  3. 当使用 super 调用时,会使用 objc_msgSendSuper 函数:id objc_msgSendSuper(struct objc_super *super, SEL op, ...)第一个参数是个objc_super的结构体,第二个参数还是类似上面的类方法的selector,
struct objc_super {
      id receiver;
      Class superClass;
};

当编译器遇到  [super setName:] 时,开始做这几个事:

1)构 建 objc_super 的结构体,此时这个结构体的第一个成员变量 receiver 就是 子类,和 self 相同。而第二个成员变量 superClass 就是指父类
调用 objc_msgSendSuper 的方法,将这个结构体和 setName 的 sel 传递过去。
2)函数里面在做的事情类似这样:从 objc_super 结构体指向的 superClass 的方法列表开始找 setName 的 selector,找到后再以 objc_super->receiver 去调用这个 selector

 

self和super之间的区别

标签:

原文地址:http://www.cnblogs.com/anancocoa/p/5059816.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!