菜鸟教程小白 发表于 2022-12-12 22:30:45

ios - 在运行时创建 UIApplicationDelegate 类


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

<p>到目前为止我的步骤是:</p>

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

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

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

<p>谁能帮我找出为什么屏幕一直黑屏?</p>

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

<p>这是我正在使用的代码:</p>

<pre><code>#import &lt;UIKit/UIKit.h&gt;
#import &lt;objc/runtime.h&gt;
#import &lt;stdio.h&gt;


id getter(id self, SEL _cmd) {
    Ivar ivar = class_getInstanceVariable(objc_getClass(&#34;AppDelegate&#34;), &#34;_window&#34;);
    id var = object_getIvar(self, ivar);
    printf(&#34;is nil: %s\n&#34;, var == nil ? &#34;true&#34; : &#34;false&#34;);
    if (var != nil) {
      printf(&#34;%s\n&#34;, [ UTF8String]);
    }
    return object_getIvar(self, ivar);
}

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

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


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

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

      class_addMethod(delegate, @selector(window), (IMP) getter, &#34;@@:&#34;);

      class_addMethod(delegate, @selector(setWindow:), (IMP) setter, &#34;v@:@&#34;);

      class_addMethod(delegate, @selector(application:didFinishLaunchingWithOptions:),
                        (IMP) didFinishLaunching, &#34;B@:@&#34;);
      objc_registerClassPair(delegate);

      return UIApplicationMain(argc, argv, nil, @&#34;AppDelegate&#34;);
    }
}
</code></pre>

<p>这是标准输出:</p>

<pre><code>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;
}
</code></pre></p>
                                    <br><hr><h1><strong>Best Answer-推荐答案</ strong></h1><br>
                                            <p><p>发现问题:ARC 正在释放我的 UIWindow。</p>

<p>将 setter 更改为</p>

<pre><code>void setter(id self, SEL _cmd, id new) {
    Ivar ivar = class_getInstanceVariable(object_getClass(self), &#34;_window&#34;);
    id old = object_getIvar(self, ivar);
    if (!) {
      if(old != nil) {
            objc_msgSend(old, sel_getUid(&#34;release&#34;));
      }
      object_setIvar(self, ivar, new);
      objc_msgSend(new, sel_getUid(&#34;retain&#34;));
    }
}
</code></pre>

<p>解决了这个问题。</p></p>
                                   
                                                <p style="font-size: 20px;">关于ios - 在运行时创建 UIApplicationDelegate 类,我们在Stack Overflow上找到一个类似的问题:
                                                        <a href="https://stackoverflow.com/questions/39947685/" rel="noreferrer noopener nofollow" style="color: red;">
                                                                https://stackoverflow.com/questions/39947685/
                                                        </a>
                                                </p>
                                       
页: [1]
查看完整版本: ios - 在运行时创建 UIApplicationDelegate 类