• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

ios - 在运行时创建 UIApplicationDelegate 类

[复制链接]
菜鸟教程小白 发表于 2022-12-12 22:30:45 | 显示全部楼层 |阅读模式 打印 上一主题 下一主题

使用 Objective C 运行时,我试图在运行时为我的 iOS 应用程序创建一个 AppDelegate。这只是为了研究目的,我无意发布。

到目前为止我的步骤是:

  • 在运行时创建一个名为 AppDelegate 的类。
  • 添加一个名为 _window 的实例变量。
  • 添加一个名为 window 的属性,该属性使用实例变量作为支持变量,并使用两个 C 函数作为 getter 和 setter。
  • 为选择器 application:didFinishLaunchingWithOptions: 添加一个方法,并在 C 中实现返回 YES。此时,该类实现了 UIApplicationDelegate 协议(protocol)。
  • 向运行时注册类。

但是,当我在连接的 iPhone 上启动该程序时,尽管程序实际上并没有崩溃,但屏幕仍然是黑色的。调试器显示我的实现被调用。按照文档,操作系统首先检查我的 window 属性是否为 nil (它是),然后创建一个 UIWindow 本身并使用我的 setter 将 UIWindow 分配给我的委托(delegate)实例。

当我访问此窗口时,它似乎功能齐全:它具有通常的边界(NSRect:Height = 667;Width = 375;X = 0;Y = 0;)并且我的自定义 ViewController 被启动,viewDidLoad 被调用。

谁能帮我找出为什么屏幕一直黑屏?

调用 [self.window makeKeyAndVisible] 不起作用,屏幕保持黑色。如果我自己在 application:didFinishLaunchingWithOptions: 中创建一个窗口并分配前一个 UIWindow 的 rootViewController,那就可以了。所以肯定是我的委托(delegate)中的 UIWindow 以某种方式损坏了。

这是我正在使用的代码:

#import <UIKit/UIKit.h>
#import <objc/runtime.h>
#import <stdio.h>


id getter(id self, SEL _cmd) {
    Ivar ivar = class_getInstanceVariable(objc_getClass("AppDelegate"), "_window");
    id var = object_getIvar(self, ivar);
    printf("is nil: %s\n", var == nil ? "true" : "false");
    if (var != nil) {
        printf("%s\n", [[NSString stringWithFormat"%@", CGRectCreateDictionaryRepresentation(((UIWindow *) var).bounds)] UTF8String]);
    }
    return object_getIvar(self, ivar);
}

void setter(id self, SEL _cmd, id new) {
    printf("setter...\n");
    Ivar ivar = class_getInstanceVariable(objc_getClass("AppDelegate"), "_window");
    object_setIvar(self, ivar, new);
}

BOOL didFinishLaunching(id self, SEL _cmd, id launchOptions) {
    printf("didFinishLaunching called\n");
    return YES;
}


int main(int argc, char * argv[]) {
    @autoreleasepool {
        Class delegate = objc_allocateClassPair([NSObject class], "AppDelegate", 0);
        class_addIvar(delegate, "_window", sizeof(UIWindow *), rint(log2(sizeof(UIWindow *))), @encode(UIWindow *));

        objc_property_attribute_t type = { "T", "@\"UIWindow\"" };
        objc_property_attribute_t strength = { "&", "" };
        objc_property_attribute_t atomic = { "N", "" };
        objc_property_attribute_t backingVar = { "V", "_window" };
        objc_property_attribute_t attrs[] = { type, strength, atomic, backingVar };
        class_addProperty(delegate, "window", attrs, 4);

        class_addMethod(delegate, @selector(window), (IMP) getter, "@@:");

        class_addMethod(delegate, @selector(setWindow, (IMP) setter, "v@");

        class_addMethod(delegate, @selector(application:didFinishLaunchingWithOptions,
                        (IMP) didFinishLaunching, "B@");
        objc_registerClassPair(delegate);

        return UIApplicationMain(argc, argv, nil, @"AppDelegate");
    }
}

这是标准输出:

is nil: true
setter...
didFinishLaunching called
is nil: false
{
    Height = 667;
    Width = 375;
    X = 0;
    Y = 0;
}
is nil: false
{
    Height = 667;
    Width = 375;
    X = 0;
    Y = 0;
}
ViewController.viewDidLoad() called
is nil: false
{
    Height = 667;
    Width = 375;
    X = 0;
    Y = 0;
}



Best Answer-推荐答案


发现问题:ARC 正在释放我的 UIWindow。

将 setter 更改为

void setter(id self, SEL _cmd, id new) {
    Ivar ivar = class_getInstanceVariable(object_getClass(self), "_window");
    id old = object_getIvar(self, ivar);
    if (![old isEqual: new]) {
        if(old != nil) {
            objc_msgSend(old, sel_getUid("release"));
        }
        object_setIvar(self, ivar, new);
        objc_msgSend(new, sel_getUid("retain"));
    }
}

解决了这个问题。

关于ios - 在运行时创建 UIApplicationDelegate 类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39947685/

回复

使用道具 举报

懒得打字嘛,点击右侧快捷回复 【右侧内容,后台自定义】
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

关注0

粉丝2

帖子830918

发布主题
阅读排行 更多
广告位

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap