NSMutableArrays为null

时间:2012-12-30 21:11:52

标签: objective-c xcode cocoa nsmutablearray nsviewcontroller

我有一个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>"
)

我可以点击视图上的一个按钮,它会调用相同的日志功能,然后吐出来:

2012-12-30 15:07:06.071 Project Land III[2177:303] (null), (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;
}

3 个答案:

答案 0 :(得分:1)

  

在我的-init方法中,两个数组都是使用+new创建的,每个数组都有一个对象。在那之后,我打电话给NSLog(@"%@, %@", entities, tiles);,它给了我正常的预期:

2012-12-30 15:07:04.160 Project Land III[2177:303] (
    "<RBEntity: 0x100508170>"
), (
    "<RBTile: 0x100508470>"
)
     

我可以点击视图上的一个按钮,它会调用相同的日志功能,然后吐出来:

2012-12-30 15:07:06.071 Project Land III[2177:303] (null), (null)

这是一个非常常见的新手错误。

您有两个 RBMainViewController对象。其中一个,你可能是用[[RBMainViewController alloc] init]之类的东西在你的其他.m文件的代码中创建的。另一个是你用笔尖创建的,可能是将它拖入笔尖。

(注意:你创建VC的nib是而不是 VC的nib。这将是循环的,让VC的nib包含正在加载nib的VC。这个VC不会它的数组驻留在nib A中,每个VC都会加载nib B。)

您在笔尖中创建的VC是其视图显示在屏幕上的VC。因为该对象从未收到init消息(它是用其他initWith…消息初始化的),所以您从未创建过它的数组。您使用init创建的视图控制器(具有其数组)在屏幕上不可见(否则您将单击其按钮而不是其他按钮,并且您已在输出中看到了数组)。

解决方案涉及两项变更。

首先,将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#>];
}

您还需要删除两个视图控制器之一。我们需要看到代码和笔尖知道哪些。如果删除在代码中创建的那个,则可能需要在该类中创建一个插座并将其连接到nib中。如果删除笔尖中的那个,你在该nib中为该VC建立的任何插座连接,则需要在代码中重新创建。

答案 1 :(得分:0)

我相信在您班上使用这些对象时,您需要使用self.entitesself.tiles

编辑(添加界面后)

您需要在界面中添加以下内容:

@interface RBMainViewController : NSViewController{
    NSMutableArray* _entities;
    NSMutableArray* _tiles;
}

@property (retain) NSMutableArray* entities;
@property (retain) NSMutableArray* tiles;

- (IBAction)log:(id)sender;

@end

然后您需要将其添加到实现中:

@synthesize entities = _entities;
@synthesize tiles = _tiles;

答案 2 :(得分:0)

为其中一个属性编写自定义getter / setter,并在那里放置一个断点以查看谁正在重置该值。如果这不能解决问题,并且您的属性仍在重置,那么您可能正在从NSViewController的{​​{1}}实例访问它们,而这些实例尚未由init初始化,因此这些属性总是没有初始化。