gpt4 book ai didi

iphone - NIB 文件中的子类 UIView 未键入子类

转载 作者:搜寻专家 更新时间:2023-10-30 19:45:05 25 4
gpt4 key购买 nike

我有一个 NIB 文件,其中一些类 (SomeClassView : UIView) 设置为 NIB 顶级 View 的自定义类。 SomeClass 有 IBOutlets,我用它来连接我在 Interface Builder 中布置的 SomeClass 的 subview 。

我像这样实例化 SomeClass:

- (id)initWithFrame:(CGRect)frame {
self = [[[[NSBundle mainBundle] loadNibNamed:@"SomeClassView" owner:nil options:nil] objectAtIndex:0] retain];
// "SomeClassView" is also the name of the nib
if (self != nil) {
self.frame = frame;
}
return self;
}

现在假设我用 SubClassView 子类化 SomeClass。我向 SubClassView 添加了一个名为 -(void)foo 的方法:

- (id)initWithFrame:(CGRect)frame {
self = [super initWithFrame:frame];
if (self != nil) {
[self foo];
}
return self;
}

此时我收到一个运行时错误:* Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[SomeClassView foo:]: unrecognized selector sent to instance 0xAAAAAA'

似乎 SubClassView 的 initWithFrame 中的“self”仍然设置为 super SomeClassView。解决此问题的快速破解方法是更改​​ SubClassView initWithFrame 中的 isa 指针:

- (id)initWithFrame:(CGRect)frame {
Class prevClass = [self class];
self = [super initWithFrame:frame];
if (self != nil) {
isa = prevClass;
[self foo]; // works now
}
}

这不是一个理想的解决方法,因为我每次子类化时都必须更新 isa,或者甚至可能有不同的 init 方法,我也必须更新这些方法。

1) 理想情况下,是否有一种简单的方法来解决这个问题,纯粹是通过代码?也许用我设置 self = 加载的 Nib 的方式?

2) 是否有替代架构更适合我正在尝试做的事情?一个示例是将所有者设置为自己,但是您必须手动设置所有属性/导出映射。

最佳答案

如果您的子类具有不是在 SomeClassView 中声明的实例变量,则交换 isa 指针是一个问题。请注意,您的 nib 文件有一个 SomeClassView 类型的对象,这意味着,在加载 nib 文件时,nib 加载程序将分配一个该类型的对象并将其从 nib 文件中解码。将 isa 指针更改为 [SubViewClass class] 暂时不会使其成为 SubViewClass 类型的对象本身,因为 nib 加载程序分配的是 SomeClassView对象。

也就是说,我认为没有一种可靠且自动的方法来使用包含在 nib 加载时需要更改其类型的对象的 nib 文件。

您可以做的是让您的 SomeClassView 对象声明一个符合某些协议(protocol)的委托(delegate)。该协议(protocol)将为 SomeClassView 中的行为定义可能被扩展的方法。例如,

@protocol SomeClassViewDelegate
@optional
- (void)someClassViewDidAwakeFromNib:(SomeClassView *)someClassView;
@end

不是继承 SomeClassView,而是让任意对象执行您当前在 SubClassView 中拥有的任何自定义行为。例如,

@interface SubClassViewBehaviour : NSObject <SomeClassViewDelegate>

@end

@implementation SubClassViewBehaviour
- (void)someClassViewDidAwakeFromNib:(SomeClassView *)someClassView {
// whatever behaviour is currently in -[SubClassView foo]
}
@end

SubClassViewBehaviour 对象将在代码中创建,并在加载 nib 文件时设置为 nib 文件的所有者,或与此相关的任何其他 IB 代理对象。 SomeClassView 将有一个委托(delegate)导出连接到文件的所有者/代理对象,它会在适当的地方调用委托(delegate)方法。例如,

@implementation SomeClassView
- (void)awakeFromNib {
SEL didAwakeFromNib = @selector(someClassViewDidAwakeFromNib:);
if ([[self delegate] respondsToSelector:didAwakeFromNib]) {
[[self delegate] performSelector:didAwakeFromNib withObject:self];
}
}
@end

进一步说明:您的代码当前泄漏了一个 View 对象,因为正在实例化两个对象:一个是通过代码中的 +alloc 实例化的,另一个是通过 nib 加载实例化的。您将后者分配给 self,因此通过 +alloc 创建的那个正在泄漏。另外,我相信您在第三个代码片段中错过了对 super 的调用。

关于iphone - NIB 文件中的子类 UIView 未键入子类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5547649/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com