我有一个 NSViewController 子类:
@property (retain) NSMutableArray* entities;
@property (retain) NSMutableArray* tiles;
在我的-init
方法,两个数组都是用 +new
创建的,并且每个被给予一个对象。之后我调用NSLog(@"%@, %@", entities, tiles);
,它给了我正如预期的那样:
2012-12-30 15:07:04.160 Project Land III[2177:303] (
"<RBEntity: 0x100508170>"
), (
"<RBTile: 0x100508470>"
)
不过,我可以单击 View 上的按钮,它会调用相同的日志函数,并且会输出以下内容:
2012-12-30 15:07:06.071 Project Land III[2177:303] (null), (null)
我已经以某种形式陷入这个问题好几天了。为什么数组是空的?
我非常乐意发布更多代码,请告诉我!
界面:
#import <Cocoa/Cocoa.h>
#import "RBEntity.h"
#import "RBTile.h"
@interface RBMainViewController : NSViewController {
NSMutableArray* _entities;
NSMutableArray* _tiles;
}
@property (retain) NSMutableArray* entities;
@property (retain) NSMutableArray* tiles;
- (IBAction)log:(id)sender;
@end
我的-init
方法:
- (id)init {
self = [super init];
self.entities = [NSMutableArray new];
self.tiles = [NSMutableArray new];
[self.entities addObject:[RBEntity entityWithLocation:NSMakePoint(4, 5) type:FACEEATER]];
[self.tiles addObject:[RBTile tileWithLocation:NSMakePoint(10, 2) type:GRASS]];
NSLog(@"%@, %@", self.entities, self.tiles);
return self;
}
最佳答案
In my
-init
method, both arrays are created with+new
, and are given one object each. After that, I callNSLog(@"%@, %@", entities, tiles);
, and it gives me just as expected:2012-12-30 15:07:04.160 Project Land III[2177:303] ( "<RBEntity: 0x100508170>" ), ( "<RBTile: 0x100508470>" )
I can click a button on the view, though, which calls the same log function, and it spit out this:
2012-12-30 15:07:06.071 Project Land III[2177:303] (null), (null)
这是一个非常常见的新手错误。
您有两个 RBMainViewController 对象。其中之一可能是您在其他 .m 文件之一的代码中创建的,方法是使用 [[RBMainViewController alloc] init]
之类的内容。另一个是您在 Nib 中创建的,可能是通过将其拖到 Nib 中来创建的。
(注意:您在其中创建 VC 的 Nib 不是 VC 的 Nib 。这将是循环的,让 VC 的 Nib 包含正在加载 Nib 的 VC。不存在的 VC其数组不会驻留在 Nib A 中,每个 VC 都会加载 Nib B。)
您在 Nib 中创建的 VC 的 View 会显示在屏幕上。因为该对象从未收到 init
消息(它是用其他一些 initWith...
消息初始化的),所以您从未创建它的数组。您使用 init
创建的 View Controller (确实有其数组)在屏幕上不可见(否则您会单击它的按钮,而不是其他按钮,并且您会看到数组在输出中)。
该解决方案涉及两个更改。
第一个是将 init
的实现更改为 initWithNibName:bundle:
的实现。就像这样:
- (instancetype) initWithNibName:(NSString *)nibName
bundle:(NSBundle *)bundle
{
self = [super initWithNibName:nibName bundle:bundle];
if (self != nil) {
<#...#>
}
return self;
}
如果您想继续使用 init
在其他代码中创建 VC,没问题,但您的 -[RBMainViewController init]
实现应该简单地发送 initWithNibName :bundle:
到 self
并返回结果。
- (instancetype) init {
return [self initWithNibName:<#nibName#> bundle:<#bundle#>];
}
您还需要删除两个 View Controller 之一。我们需要查看代码和 Nib 才能知道是哪个。如果删除在代码中创建的 socket ,您可能需要在该类中创建一个 socket 并将其连接到 Nib 。如果您删除 Nib 中的 socket ,则您在该 Nib 中与该 VC 建立的任何 socket 连接都需要在代码中重新创建。
关于objective-c - NSMutableArray 为 null,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14094701/