iOS经典讲解之[self class]和[super class]的区别

本文深入探讨了Objective-C编程语言中self和super关键字的区别,并详细解释了它们在方法调用和消息传递机制中的底层实现。通过分析实例代码,展示了如何在子类中使用self和super来调用自身或父类的方法,以及这两种方法在消息查找过程中的作用。文章还揭示了编译器如何将方法调用转换为C函数调用,以及objc_runtimeRef中提到的几种消息发送方式。

作者:Loving_iOS

转载请标明出处:http://blog.youkuaiyun.com/loving_ios/article/details/49884599

在objc中的类实现中经常看到这两个关键字”self”和”super”,但他们的区别表现在各个方面。
以下面的代码为例:

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

@end

@interface Son:Father
{
NSUInteger age;
}
- (void) setAge:(NSUInteger) age;
- (void) setName:(NSString*) yourName andAge:(NSUInteger) age;
@end
@implementation Son
- (void) setName:(NSString*) yourName andAge:(NSUInteger) age 
{
[self setAge:age];
[super setName:yourName];
}
@end
int main(int argc, char* argv[]) {
AutoreleasePool{
Son* son = [[Son alloc] init];
[son setName:@"aaa" andAge:18];

}
return 0;
}
 

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

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

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

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

self 's class is Son

super ' s class is Son

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

具体原因要看他们的内部实现机制是什么样的:

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

可以看到会转成调用上面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这个类的父类是什么,拿上面的代码为例,当编译器遇到Son里setName:andAge方法里的[super setName:]时,开始做这几个事:

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

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

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


<think> 我们正在讨论的是Python中的`__class__``super()`,以及它们之间的关系。根据引用[1],在Python3中,`super()`的实现依赖于一个名为`__class__`的变量,这个变量在方法定义时被隐式地创建,它指向的是定义该方法的类。 在Python中,`super()`函数用于调用父类(超类)的方法。这在继承中非常有用,尤其是当子类重写了父类的方法,但还想调用父类的实现时。 引用[1]提到,`super()`会从调用栈中查找`__class__`这个名字。实际上,在方法定义时,`__class__`是一个自由变量,它会被闭包捕获,指向定义该方法的类。 关于`class __super__`,在Python中并没有直接名为`__super__`的内置类或属性。可能用户是混淆了`super()``__class__`,或者是指某些特定框架中的用法。在Python中,我们通常使用`super()`来获取一个代理对象,该对象会按照方法解析顺序(MRO)来调用父类的方法。 下面是一个简单的示例,说明如何使用`super()`: ```python class Parent: def __init__(self): print("Parent __init__") class Child(Parent): def __init__(self): super().__init__() # 调用父类的__init__方法 print("Child __init__") c = Child() ``` 输出: ``` Parent __init__ Child __init__ ``` 在这个例子中,`Child`类继承自`Parent`类。在`Child`的`__init__`方法中,我们使用`super().__init__()`来调用`Parent`的`__init__`方法。 另外,引用[1]提到,可以直接使用`__class__`,例如: ```python class Child(Parent): def __init__(self): # 使用__class__super()的等价形式 super(__class__, self).__init__() print("Child __init__") ``` 这种形式中,`super(__class__, self)`显式地指定了类实例。在Python3中,通常可以省略参数,因为解释器会自动填充,但在一些复杂情况下(比如多重继承)可能需要显式指定。 需要注意的是,在类方法(`@classmethod`)中,`super()`的行为类似,但会绑定到类而不是实例。例如: ```python class Parent: @classmethod def class_method(cls): print("Parent class method") class Child(Parent): @classmethod def class_method(cls): super().class_method() # 调用父类的类方法 print("Child class method") Child.class_method() ``` 输出: ``` Parent class method Child class method ``` 总结: - `__class__`是一个在方法定义时自动创建的变量,指向定义该方法的类。 - `super()`函数用于调用父类的方法,它依赖于`__class__`当前的实例(或类,如果是类方法的话)。 - 在Python3中,通常可以无参数使用`super()`,即`super()`,解释器会自动填充为`super(__class__, self)`(在实例方法中)或`super(__class__, cls)`(在类方法中)。 关于用户提到的`class __super__`,可能是一个误解。在Python中,没有直接名为`__super__`的内置类。`super()`返回的是一个`super`对象,它是一个代理对象,用于委托方法调用。 如果用户是在其他上下文中(比如某个特定的框架)看到`__super__`,那可能是框架自己定义的。但在标准的Python中,不存在这样的名字。 因此,我们重点解释`super()``__class__`的关系用法。 §§相关问题§§ 1. 在多重继承中,`super()`是如何工作的? 2. 在Python2Python3中,`super()`的使用有什么不同? 3. 如何理解方法解析顺序(MRO)? 4. 在静态方法中可以使用`super()`吗? </think>### `__class__``super()`的概念及用法 在Python中,`__class__``super()`是与类继承方法调用相关的关键概念,主要用于在子类中访问父类的属性方法。 #### 1. `__class__`的含义 `__class__`是一个**隐式闭包变量**,在类方法定义时自动创建: - 指向**当前方法被定义的类**(不是调用时的类) - 在方法内部可直接访问:`__class__` - 常用于与`super()`配合调用父类方法 示例: ```python class Parent: def show(self): print("Parent method") class Child(Parent): def show(self): print("Child method") # __class__指向Child,即使通过子类实例调用 print("Defined in:", __class__.__name__) # 输出:Child ``` #### 2. `super()`的工作原理 `super()`返回父类的**代理对象**,用于调用父类方法: - **单参数形式**:`super(__class__, self)`(显式指定类) - **零参数形式**:`super()`(Python 3+ 推荐用法) - 通过MRO(方法解析顺序)确定调用顺序 ```python class Parent: def run(self): print("Parent running") class Child(Parent): def run(self): super().run() # 调用父类方法 print("Child running") obj = Child() obj.run() ``` 输出: ``` Parent running Child running ``` #### 3. 关键配合用法 在子类方法中组合使用`__class__``super()`: ```python class Parent: def process(self): print("Parent processing") class Child(Parent): def process(self): # 显式调用父类方法 super(__class__, self).process() print("Child processing") ``` > 引用[1]指出:`super`的实现依赖`__class__`变量,它会从调用栈中查找该方法被定义的类[^1]。 #### 4. 与iOS的对比 引用[2]提到iOS中的`[super class]`机制: - Objective-C通过`objc_msgSendSuper`结构体实现父类调用 - Python的`super()`是动态代理,而iOS是静态编译确定 - Python的`__class__`类似于Objective-C的`self->isa`指针[^2] #### 5. 最佳实践 1. **Python 3+** 优先使用零参数形式 `super()` 2. 需要明确指定类时使用 `super(__class__, self)` 3. 避免在类外部直接使用 `__class__` 4. 多重继承时注意MRO顺序(`类名.mro()`查看)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值