iOS自定义初始化方法

有很多初始化方法我们通常都是在applicationDidFinishLaunching里一个个调用,那么有没有办法像__attribute__((constructor)),能够自动调用被修饰的函数?

可以通过指定函数所在section的方式,然后获取section开头去逐个调用。但是这种方法有个问题,你只能获取到第一个函数的开头,但是你不知道它的大小,也就没有办法去依次获取了。

cpp 复制代码
__attribute__((used, section("__TEXT, CustomInit")))
void custom1() {
    NSLog(@"custom1");
}

__attribute__((used, section("__TEXT, CustomInit")))
void custom2() {
    NSLog(@"custom2");
}

- (void)callCustom
{
    Method orginalMethod = class_getClassMethod([self class], _cmd);
    IMP imp = method_getImplementation(orginalMethod);
    Dl_info info;
    if (dladdr((void *)imp, &info)) {
        printf("dli_fname: %s\n", info.dli_fname);
        printf("dli_sname: %s\n", info.dli_sname);
        printf("dli_fbase: %p\n", info.dli_fbase);
        printf("dli_saddr: %p\n", info.dli_saddr);
    } else {
        printf("error: can't find that symbol.\n");
    }
    
    unsigned long funcSize = 0;
    //CustomFunc段的开始
    uint8_t *funcStart = getsectiondata((struct mach_header_64 *)info.dli_fbase, "__TEXT", "CustomInit", &funcSize);
    ((void(*)())funcStart)();
}

所以可以通过声明全局或者静态指针变量,去持有这些函数,把这些指针放到__DATA里自定义的一个段里,因为指针大小都是8字节,每次跳8字节就能访问到下一个指针了。

cpp 复制代码
void custom1() {
    NSLog(@"custom1");
}

void(*customVar1)(void) __attribute__((used, section("__TEXT, CustomInit")))  = custom1;

void custom2() {
    NSLog(@"custom2");
}

void(*customVar2)(void) __attribute__((used, section("__TEXT, CustomInit")))  = custom1;


- (void)callCustom
{
    Method orginalMethod = class_getClassMethod([self class], _cmd);
    IMP imp = method_getImplementation(orginalMethod);
    Dl_info info;
    if (dladdr((void *)imp, &info)) {
        printf("dli_fname: %s\n", info.dli_fname);
        printf("dli_sname: %s\n", info.dli_sname);
        printf("dli_fbase: %p\n", info.dli_fbase);
        printf("dli_saddr: %p\n", info.dli_saddr);
    } else {
        printf("error: can't find that symbol.\n");
    }
    unsigned long size = 0;
    uint8_t *start = getsectiondata((struct mach_header_64 *)info.dli_fbase, "__DATA", "CustomInit", &size);
        int funcCount = size/sizeof(void *);
    for (int i=0;i<funcCount; i++) {
        //这里要用指针的指针
        void(**f)() = (void(**)())start + i;
        (*f)();
    }
}

这样声明函数和变量太麻烦了,我们可以使用宏来帮忙。

cpp 复制代码
#define CustomInitFuncBegin(funcName) \
void funcName() {

#define CustomInitFuncEnd(funcName) }\
static void (*funcName##_var)() __attribute__((used, section("__DATA, CustomInit")))  = funcName;


CustomInitFuncBegin(init1)
NSLog("测试init1\n");
CustomInitFuncEnd(init1)

CustomInitFuncBegin(init2);
NSLog("测试init2\n");
CustomInitFuncEnd(init2);


+ (void)callCustom
{
    Method orginalMethod = class_getClassMethod([self class], _cmd);
    IMP imp = method_getImplementation(orginalMethod);
    Dl_info info;
    if (dladdr((void *)imp, &info)) {
        printf("dli_fname: %s\n", info.dli_fname);
        printf("dli_sname: %s\n", info.dli_sname);
        printf("dli_fbase: %p\n", info.dli_fbase);
        printf("dli_saddr: %p\n", info.dli_saddr);
    } else {
        printf("error: can't find that symbol.\n");
    }
    unsigned long size = 0;
    uint8_t *start = getsectiondata((struct mach_header_64 *)info.dli_fbase, "__DATA", "CustomInit", &size);
    int funcCount = size/sizeof(void *);
    for (int i=0;i<funcCount; i++) {
        void(**f)() = (void(**)())start + i;
        (*f)();
    }
}

参考:

https://everettjf.github.io/2017/03/06/a-method-of-delay-premain-code/

相关推荐
2501_9159090615 分钟前
iOS App 上架全流程详解:证书配置、打包上传、审核技巧与跨平台上架工具 开心上架 实践
android·ios·小程序·https·uni-app·iphone·webview
2501_9151063219 分钟前
iOS 26 系统流畅度测试实战分享,多工具组合辅助策略
android·macos·ios·小程序·uni-app·cocoa·iphone
2501_9159184119 分钟前
开发 iOS 应用全流程指南,环境搭建、证书配置与跨平台使用 开心上架 上架AppStore
android·ios·小程序·https·uni-app·iphone·webview
方君宇28 分钟前
iOS App小组件(Widget)显示LottieFiles动画和GIF图片
ios
Digitally37 分钟前
如何将 iPhone 联系人同步到 Mac
macos·ios·iphone
Digitally39 分钟前
无需 iCloud 在 iPhone 之间传输文本消息
ios·iphone·icloud
2501_915921439 小时前
iOS 是开源的吗?苹果系统的封闭与开放边界全解析(含开发与开心上架(Appuploader)实战)
android·ios·小程序·uni-app·开源·iphone·webview
2501_9159090612 小时前
原生 iOS 开发全流程实战,Swift 技术栈、工程结构、自动化上传与上架发布指南
android·ios·小程序·uni-app·自动化·iphone·swift
2501_9151063212 小时前
Comodo HTTPS 在工程中的部署与排查实战(证书链、兼容性与真机抓包策略)
网络协议·http·ios·小程序·https·uni-app·iphone
2501_9159090612 小时前
苹果软件混淆与 iOS 代码加固趋势,IPA 加密、应用防反编译与无源码保护的工程化演进
android·ios·小程序·https·uni-app·iphone·webview