iOS Runtime的工具类

2022-05-26  本文已影响0人  心猿意码_

QMRuntimeHelper.h


#import <Foundation/Foundation.h>

typedef void(^RuntimeHookBlock)(id ,NSArray *);

@interface QMRuntimeHelper : NSObject

/**
 *  用一个新的方法替换类中的一个已经存在的方法
 *
 *  @param aclass            要修改的类
 *  @param originalSelector 要替换的方法的选择器
 *  @param swizzledSelector 新方法的选择器
 *  @param isInstanceMethod 是否实例方法
 */
+ (void)setMethodSwizzlingForClass:(Class)aclass originalSelector:(SEL)originalSelector swizzledSelector:(SEL)swizzledSelector isInstanceMethod:(BOOL)isInstanceMethod;

/**
 *  在运行时判断类中是否包含某个属性
 *
 *  @param aclass        类
 *  @param propertyName 属性名
 *
 *  @return 该属性是否存在
 */
+ (BOOL)isClass:(Class)aclass hasProperty:(NSString*)propertyName;

/**
 *  在运行时判断类中是否包含某个实例方法,类似于NSObject的instanceRespondsToSelector
 *
 *  @param aclass              类
 *  @param instanceMethodName 方法名
 *
 *  @return 该方法是否存在
 */
+ (BOOL)isClass:(Class)aclass hasInstanceMethod:(NSString*)instanceMethodName;

/**
 *  在运行时判断类中是否包含某个类方法
 *
 *  @param aclass           类
 *  @param classMethodName 方法名
 *
 *  @return 该方法是否存在
 */
+ (BOOL)isClass:(Class)aclass hasClassMethod:(NSString *)classMethodName;

/**
 *  在运行时判断一个类是否存在
 *
 *  @param className 类名
 *
 *  @return 是否存在
 */
+ (BOOL)isClassExist:(NSString *)className;

/**
 *  修改类中的一个方法,往里面添加一些代码(本质上是方法置换)
 *
 *  @param aclass              要修改的类
 *  @param originalMethodName 要修改的方法名
 *  @param isInstanceMethod   是否实例方法
 *  @param block              增加的代码
 *
 *  @return 是否添加-成功
 */
+ (BOOL)hookMethodForClass:(Class)aclass originalMethod:(NSString*)originalMethodName isInstanceMethod:(BOOL)isInstanceMethod hookedBlock:(RuntimeHookBlock)block;

/**
 *  获取类中某个方法的实现block
 *
 *  @param aclass          类
 *  @param methodSelector 方法选择器
 *
 *  @return 该方法对应的block
 */
+ (id)methodBlockForClass:(Class)aclass method:(SEL)methodSelector;

/**
 *  获取所有遵循了指定协议的类
 *
 *  @param protocol 要遵循的协议
 *
 *  @return 遵循了协议的类
 */
+ (NSArray *)getClassesThatConfirmToProtocol:(Protocol*)protocol;

@end

QMRuntimeHelper.m


#import "QMRuntimeHelper.h"
#import <objc/runtime.h>

@implementation QMRuntimeHelper

+(void)setMethodSwizzlingForClass:(Class)class originalSelector:(SEL)originalSelector swizzledSelector:(SEL)swizzledSelector isInstanceMethod:(BOOL)isInstanceMethod {
    Method originalMethod;
    Method swizzledMethod;
    
    if (isInstanceMethod) {
        originalMethod = class_getInstanceMethod(class, originalSelector);
        swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
    } else {
        originalMethod = class_getClassMethod(class, originalSelector);
        swizzledMethod = class_getClassMethod(class, swizzledSelector);
    }

 
    BOOL didAddMethod =
    class_addMethod(class,
                    originalSelector,
                    method_getImplementation(swizzledMethod),
                    method_getTypeEncoding(swizzledMethod));
    
    if (didAddMethod) {
        class_replaceMethod(class,
                            swizzledSelector,
                            method_getImplementation(originalMethod),
                            method_getTypeEncoding(originalMethod));
    } else {
        method_exchangeImplementations(originalMethod, swizzledMethod);
    }
}

+ (BOOL)isClass:(Class)class hasProperty:(NSString *)propertyName {
    return (class_getProperty(class, [propertyName UTF8String]) != NULL);
}

+ (void)printPropertiesOfClass:(Class)class {
    unsigned int count = 0;
    objc_property_t *propList = class_copyPropertyList(class, &count);
    for (int i=0; i<count; i++) {
        objc_property_t property = propList[i];
        const char* char_f =property_getName(property);
        NSString *propertyName = [NSString stringWithUTF8String:char_f];
        MYLog(@"proprty = %@",propertyName);
    }
    free(propList);

}


+(BOOL)isClass:(Class)class hasInstanceMethod:(NSString*)instanceMethodName {
    SEL selector = NSSelectorFromString(instanceMethodName);
    Method method = class_getInstanceMethod(class, selector);
    return (method != NULL);
}

+(BOOL)isClass:(Class)class hasClassMethod:(NSString *)classMethodName {
    SEL selector = NSSelectorFromString(classMethodName);
    Method method = class_getClassMethod(class,selector);
    return (method != NULL);
}

+ (BOOL)isClassExist:(NSString *)className {
    Class class = objc_getClass([className UTF8String]);
    return (class!=nil);
}

+ (BOOL)hookMethodForClass:(Class)class originalMethod:(NSString*)originalMethodName isInstanceMethod:(BOOL)isInstanceMethod hookedBlock:(RuntimeHookBlock)block {
    NSString *hookedMethodName = [NSString stringWithFormat:@"hook%@",originalMethodName];
    SEL hookedSelector = NSSelectorFromString(hookedMethodName);
    
    Method originalMethod;
    if (isInstanceMethod) {
        originalMethod = class_getInstanceMethod(class, NSSelectorFromString(originalMethodName));
    } else {
        originalMethod = class_getClassMethod(class, NSSelectorFromString(originalMethodName));
    }
  
    //return type
    char* returnType = (char*)malloc(sizeof(char));
    returnType = method_copyReturnType(originalMethod);
//    printf("returnType %s ", returnType);
    //argument type
    unsigned int argumentCount = method_getNumberOfArguments(originalMethod);
    //最多保存4个参数,前面两个固定的,后面是可变的
    char* argumentType = malloc(sizeof(char)*4);
    memset(argumentType, 0, 4);
    for (unsigned int i = 0; i<argumentCount; i++) {
        method_getArgumentType(originalMethod, i, &argumentType[i], 1);
    }
    
    IMP imp = NULL;
    char *argTypes = malloc(sizeof(char) * (argumentCount+1));
    memset(argTypes, 0, argumentCount +1);
    argTypes[0] = *returnType;
//    argTypes = strcpy(argTypes, returnType);
    if (argumentCount == 2) {
        imp = imp_implementationWithBlock(^id (id _self) {
            id object = class;
            if (isInstanceMethod) {
                object = _self;
            }
            id returnValue = [QMRuntimeHelper performHookedMethod:hookedSelector withArgs:@[] forObject:object isInstanceMethod:isInstanceMethod];
            if (block) {
                block(_self, @[]);
            }

            return returnValue;
        });
        argTypes = strcat(argTypes, argumentType);

    } else if (argumentCount == 3) {
        imp = imp_implementationWithBlock(^id (id _self, id arg1){
            id returnValue;
            id object = class;
            if (isInstanceMethod) {
                object = _self;
            }
            if (arg1) {
                returnValue = [QMRuntimeHelper performHookedMethod:hookedSelector withArgs:@[arg1] forObject:object isInstanceMethod:isInstanceMethod];
                if (block) {
                    block(_self, @[arg1]);
                }
            } else {
                returnValue = [QMRuntimeHelper performHookedMethod:hookedSelector withArgs:@[] forObject:object isInstanceMethod:isInstanceMethod];
                if (block) {
                    block(_self, @[]);
                }
            }

            return returnValue;
        });
        argTypes = strcat(argTypes, argumentType);

    } else if (argumentCount == 4) {
        imp = imp_implementationWithBlock(^id (id _self, id arg1, id arg2){
            id object = class;
            if (isInstanceMethod) {
                object = _self;
            }
            id returnValue = [QMRuntimeHelper performHookedMethod:hookedSelector withArgs:@[arg1,arg2] forObject:object isInstanceMethod:isInstanceMethod];;

            if (block) {
                block(_self,@[arg1,arg2]);
            }
            return returnValue;
        });
        argTypes = strcat(argTypes, argumentType) ;


    }
    
    IMP originImp = method_getImplementation(originalMethod);
    BOOL succ;
    if (isInstanceMethod) {
        succ = class_addMethod(class, hookedSelector, originImp, argTypes);
    } else {
        succ = class_addMethod(object_getClass(class), hookedSelector, originImp, argTypes);
    }
    
    free(argTypes);
    free(returnType);
    method_setImplementation(originalMethod, imp);
//    free(returnType);
    return succ;
}

+ (id)performHookedMethod:(SEL)hookedSelector withArgs:(NSArray*)args forObject:(id)object isInstanceMethod:(BOOL)isInstanceMethod {
    id returnValue;
//    id arg1,arg2;
//    if (args.count >0) {
//        arg1 = args[0];
//    }
//    if (args.count >1) {
//        arg2 = args[1];
//    }
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
   
    if ([object respondsToSelector:hookedSelector]) {
        returnValue =[object performSelector:hookedSelector];
    }
   
#pragma clang diagnostic pop
    return returnValue;
}

+ (id)methodBlockForClass:(Class)class method:(SEL)methodSelector {
    IMP imp = class_getMethodImplementation(class, methodSelector);
    return imp_getBlock(imp);
}

+ (NSArray *)getClassesThatConfirmToProtocol:(Protocol *)protocol {
    NSMutableArray *classes = [NSMutableArray array];
    unsigned int classCount;
    Class* classList = objc_copyClassList(&classCount);

    int i;
    for (i=0; i<classCount; i++) {
        const char *className = class_getName(classList[i]);
        Class thisClass = objc_getClass(className);
        if (class_conformsToProtocol(thisClass, protocol)) {
            [classes addObject:thisClass];
        }
    }
    free(classList);
    return classes;
}

@end


上一篇 下一篇

猜你喜欢

热点阅读