美文网首页runtime.runloop
常用的runtime方法交换

常用的runtime方法交换

作者: Iamyu | 来源:发表于2021-03-23 16:12 被阅读0次

一、用于记录控制器事件

其实可以在控制器的viewWillAppear、viewDidLoad等方法中添加追踪代码来实现,但是需要很多的重复代码;使用继承同样的也会有不少重复代码。如果用分类来进行方法交换:

    // 通过class_getInstanceMethod()函数从当前对象中的method list获取method结构体,如果是类方法就使用class_getClassMethod()函数获取。

        Method fromMethod = class_getInstanceMethod([self class], @selector(viewDidLoad));

        MethodtoMethod =class_getInstanceMethod([selfclass],@selector(swizzlingViewDidLoad));

        /**

         *  我们在这里使用class_addMethod()函数对Method Swizzling做了一层验证,如果self没有实现被交换的方法,会导致失败。

         *  而且self没有交换的方法实现,但是父类有这个方法,这样就会调用父类的方法,结果就不是我们想要的结果了。

         *  所以我们在这里通过class_addMethod()的验证,如果self实现了这个方法,class_addMethod()函数将会返回NO,我们就可以对其进行交换了。

         */

        if (!class_addMethod([self class], @selector(swizzlingViewDidLoad), method_getImplementation(toMethod), method_getTypeEncoding(toMethod)))   {

            method_exchangeImplementations(fromMethod, toMethod);

        }

二、防止按钮重复点击

创建按钮的分类

.声明文件(.h)

#define defaultInterval .5//默认时间间隔

@interface UIButton (UIbutton_Delay)

@property(nonatomic,assign)NSTimeInterval timeInterval;//用这个给重复点击加间隔

@property(nonatomic,assign)BOOL isIgnoreEvent;//YES不允许点击NO允许点击

@end

.实现文件(.m)

#import <objc/runtime.h>

@implementation UIButton (UIbutton_Delay)

- (NSTimeInterval)timeInterval{

    return[objc_getAssociatedObject(self,_cmd)doubleValue];

}

- (void)setTimeInterval:(NSTimeInterval)timeInterval{

    objc_setAssociatedObject(self,@selector(timeInterval),@(timeInterval),OBJC_ASSOCIATION_RETAIN_NONATOMIC);

}

//runtime动态绑定属性

- (void)setIsIgnoreEvent:(BOOL)isIgnoreEvent{

    objc_setAssociatedObject(self,@selector(isIgnoreEvent),@(isIgnoreEvent),OBJC_ASSOCIATION_RETAIN_NONATOMIC);

}

- (BOOL)isIgnoreEvent{

    return[objc_getAssociatedObject(self,_cmd)boolValue];

}

- (void)resetState{

    [self setIsIgnoreEvent:NO];

}

+ (void)load{

    static dispatch_once_t onceToken;

    dispatch_once(&onceToken, ^{

        SEL selA =@selector(sendAction:to:forEvent:);

        SEL selB =@selector(mySendAction:to:forEvent:);

        Method methodA =class_getInstanceMethod(self, selA);

        Method methodB =class_getInstanceMethod(self, selB);

        //将methodB的实现添加到系统方法中也就是说将methodA方法指针添加成方法methodB的返回值表示是否添加成功

        BOOL isAdd =class_addMethod(self, selA,method_getImplementation(methodB),method_getTypeEncoding(methodB));

        //添加成功了说明本类中不存在methodB所以此时必须将方法b的实现指针换成方法A的,否则b方法将没有实现。

        if(isAdd) {

            class_replaceMethod(self, selB,method_getImplementation(methodA),method_getTypeEncoding(methodA));

        }else{

            //添加失败了说明本类中有methodB的实现,此时只需要将methodA和methodB的IMP互换一下即可。

            method_exchangeImplementations(methodA, methodB);

        }

    });

}

- (void)mySendAction:(SEL)action to:(id)target forEvent:(UIEvent*)event{

    if([NSStringFromClass(self.class)isEqualToString:@"UIButton"]) {

        self.timeInterval=self.timeInterval==0?defaultInterval:self.timeInterval;

        if(self.isIgnoreEvent){

            return;

        }else if(self.timeInterval>0){

            [self performSelector:@selector(resetState) withObject:nil afterDelay:self.timeInterval];

        }

    }

    //此处methodA和methodB方法IMP互换了,实际上执行sendAction;所以不会死循环

    self.isIgnoreEvent=YES;

    [self mySendAction:action to:target forEvent:event];

}

三、增大按钮的点击热区

.h 文件

#import <UIKit/UIKit.h>

@interface UIButton (HitAreaExpand)

@property (nonatomic) CGFloat minHitTestWidth;

@property (nonatomic) CGFloat minHitTestHeight;

@end

.m 文件

#import "UIButton+HitAreaExpand.h"

#import <objc/runtime.h>

@implementation UIButton (HitAreaExpand)

- (CGFloat)minHitTestWidth {

    NSNumber * width = objc_getAssociatedObject(self, @selector(minHitTestWidth));

    return [width floatValue];

}

- (void)setMinHitTestWidth:(CGFloat)minHitTestWidth {

    objc_setAssociatedObject(self, @selector(minHitTestWidth), [NSNumber numberWithFloat:minHitTestWidth], OBJC_ASSOCIATION_ASSIGN);

}

- (CGFloat)minHitTestHeight {

    NSNumber * height = objc_getAssociatedObject(self, @selector(minHitTestHeight));

    return [height floatValue];

}

- (void)setMinHitTestHeight:(CGFloat)minHitTestHeight {

    objc_setAssociatedObject(self, @selector(minHitTestHeight), [NSNumber numberWithFloat:minHitTestHeight], OBJC_ASSOCIATION_ASSIGN);

}

- (BOOL)pointInside:(CGPoint)point withEvent:(nullable UIEvent *)event {

    return CGRectContainsPoint(HitTestingBounds(self.bounds, self.minHitTestWidth, self.minHitTestHeight), point);

}

CGRect HitTestingBounds(CGRect bounds, CGFloat minimumHitTestWidth, CGFloat minimumHitTestHeight) {

    CGRect hitTestingBounds = bounds;

    if (minimumHitTestWidth > bounds.size.width) {

        hitTestingBounds.size.width = minimumHitTestWidth;

        hitTestingBounds.origin.x -= (hitTestingBounds.size.width - bounds.size.width)/2;

    }

    if (minimumHitTestHeight > bounds.size.height) {

        hitTestingBounds.size.height = minimumHitTestHeight;

        hitTestingBounds.origin.y -= (hitTestingBounds.size.height - bounds.size.height)/2;

    }

    return hitTestingBounds;

}

四、防止数组越界造成奔溃(该方法需要注意类簇)

@implementation NSArray (Limit)

+ (void)load {

    //[super load];

    //可变数组方法调换

    Method fromMethod = class_getInstanceMethod(objc_getClass("__NSArrayM"), @selector(objectAtIndexedSubscript:));

    Method toMethod = class_getInstanceMethod(objc_getClass("__NSArrayM"), @selector(cm_objectAtIndexedSubscript:));

    method_exchangeImplementations(fromMethod, toMethod);

    //不可变数组方法调换

    Method a = class_getInstanceMethod(objc_getClass("__NSArrayI"), @selector(objectAtIndex:));

    Method b = class_getInstanceMethod(objc_getClass("__NSArrayI"), @selector(cm_objectAtIndex:));

    method_exchangeImplementations(a, b);

}

- (id)cm_objectAtIndexedSubscript:(NSUInteger)index {

    // 判断下标是否越界,如果越界就进入异常拦截

    if (self.count-1 < index) {

        @try {

            return nil;

        }

        @catch (NSException *exception) {

            // 在崩溃后会打印崩溃信息

            NSLog(@"---------- %s 奔溃信息 Method  ----------\n", class_getName(self.class));

            NSLog(@"%@", [exception callStackSymbols]);

            return nil;

        }

        @finally {}

    } // 如果没有问题,则正常进行方法调用

    else {

        return [self cm_objectAtIndexedSubscript:index];

    }

}

- (id)cm_objectAtIndex:(NSUInteger)index {

    // 判断下标是否越界,如果越界就进入异常拦截

    if (self.count-1 < index) {

        @try {

            return nil;

        }

        @catch (NSException *exception) {

            // 在崩溃后会打印崩溃信息

            NSLog(@"---------- %s 奔溃信息 Method  ----------\n", class_getName(self.class));

            NSLog(@"%@", [exception callStackSymbols]);

            return nil;

        }

        @finally {}

    } // 如果没有问题,则正常进行方法调用

    else {

        return [self cm_objectAtIndex:index];

    }

}

@end

相关文章

  • runtime和oc内存区域(2018-04-02)

    runtime常用的几个方法: 交换方法 动态添加属性 动态添加方法 1.交换方法 class_getClassM...

  • Runtime应用之交换方法实现

    Runtime一个常用的场景是交换方法的调用。其实就是利用了Runtime的方法交换,具体代码如下: 核心思路是先...

  • Runtime方法的使用—Class篇

    原文来自:Runtime方法的使用—Class篇感谢:xietao3 方法交换: 常用方法

  • 常用的runtime方法交换

    一、用于记录控制器事件 其实可以在控制器的viewWillAppear、viewDidLoad等方法中添加追踪代码...

  • Runtime(一)方法交换

    Runtime的常用方法(交换方法) 举个简单的例子就是在我们调用UIImage的imageNamed方法时判断加...

  • runTime常用方法

    使用runTime改变实例成员的值 使用runtime来交换两个方法 注意再次调用该方法不交换 使用runTime...

  • runtime

    runtime交换方法 动态添加方法

  • Runtime

    runtime运行时机制1:通过runtime,实现方法交换(交换两个类方法、交换两个实例方法)2:通过runti...

  • Day3

    1 runtime运行时机制1:通过runtime,实现方法交换(交换两个类方法、交换两个实例方法)。2:通过ru...

  • runtime的理解(二)

    主要内容 利用 runtime 交换方法 利用 runtime 动态添加方法 利用 runtime 动态添加属性 ...

网友评论

    本文标题:常用的runtime方法交换

    本文链接:https://www.haomeiwen.com/subject/iaufhltx.html