iOS-在子类中调用父类私有方法

https://www.jianshu.com/p/6171e6ec78a3
三方中在.h中声明的方法比较好重写,直接在子类里重写方法,如果需要重调父类原有方法可以用super直接调用。而一些私有方法并没有在.h中声明,这时我们也可以直接重写父类方法,但是相对于想要重调父类方法,就不能用super直接调用来实现。这里super只是一个编译器修饰符,是一个指向父类标志,并不是对象的父类实例。所以你再用super调用父类的私有方法是没有作用的。这时我们有几种选择可以帮我们实现父类私有方法的调用。

首先可以用Method Swizzling(方法交换)来帮我们实现,但是此方法会引起全局的方法指针交换,多人开发中如果没有交流好很容易出现一些纰漏等问题

第二种方法是我们可以利用runtime的消息发送机制,为我们的对象调用其父类的私有方法。主要用到objc_msgSendSuper方法,但是直接调用是不安全的,因为你不能确定父类是否含有这个方法,所以在调用之前需要判断父类是否含有此方法,举个例子,代码如下:

#import "Parent.h"

@implementation Parent

- (void)say {
    NSLog(@"我是父类的私有方法say");
}

@end
#import "Children.h"
#import <objc/message.h>

@implementation Children

- (void)say {
    if ([self containsSuperMethod:@"say"]) {
        struct objc_super super_obj;
        super_obj.receiver = self;
        super_obj.super_class = [Parent class];
        void (*say)(void *, SEL) = (void *)objc_msgSendSuper;
        say(&super_obj, sel_registerName("say"));
    }
}

- (BOOL)containsSuperMethod:(NSString *)methodName {
    unsigned int outCount = 0;
    Method *methods = class_copyMethodList([Parent class], &outCount);
    for (int i = 0; i < outCount; i++) {
        Method method = methods[i];
        SEL methodNameSEL = method_getName(method);
        if ([methodName isEqualToString:NSStringFromSelector(methodNameSEL)]) {
            free(methods);
            return YES;
        }
    }
    free(methods);
    return NO;
}

这个方法是利用遍历父类中的所有方法来判断是否包含某一方法,这样做其实也有一定的弊端,比如当这个父类方法很多时,而你又需要频繁调用此方法时就会引起不必要的消耗。如果父类包含此方法就直接利用objc_msgSendSuper发送消息就可以了,需要声明#import <objc/message.h>

第三种是先获取父类方法,然后利用构建的方式来调用。判断父类是否包含此方法是根据构建出来的方法是否存在来判断的,相比于上一个方法减少了遍历的过程。例子代码如下:

- (void)say {
    Method method = class_getInstanceMethod([Parent class], sel_registerName("say"));
    void (*super_func)(id,SEL) = (void *)method_getImplementation(method);
    if (super_func){
        super_func(self, sel_registerName("say"));
    }
}

if (super_func) 是判断父类是否包含的此方法,如果含有此方法则调用,否则不调用。

参考:https://www.jianshu.com/p/6171e6ec78a3
https://www.cnblogs.com/zbblog/p/12333073.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值