使用NSWindowController的showWindow记忆泄漏:

时间:2011-05-25 14:41:19

标签: objective-c cocoa memory-leaks nswindow nswindowcontroller

我一直在努力解决以下泄漏问题。我通过Instruments将其缩小到以下代码块:

- (NewMessageWindowController *)showNewMessageWindowWithRecipients:(NSArray *)recipients {

    NewMessageWindowController * newMessageWindowController = [[NewMessageWindowController alloc] init];
    [newMessageWindowController showWindow:self]; // 100% on this line.
    [newMessageWindowController.toField setStringValue:[recipients componentsJoinedByString:@","]];
    [newMessageWindowController.messageView becomeFirstResponder];
    [windowControllers addObject:newMessageWindowController];
    [newMessageWindowController release];

    return newMessageWindowController;
}

这个块被调用如下:

[AppDelegate showNewMessageWindowWithRecipients:[NSArray arrayWithObject:recipient]];

recipient只是一个NSString。

以下是乐器的回溯:

  30 Friendz start
  29 AppKit NSApplicationMain
  28 AppKit -[NSApplication run]
  27 AppKit -[NSApplication sendEvent:]
  26 AppKit -[NSWindow sendEvent:]
  25 AppKit -[NSWindow keyDown:]
  24 AppKit forwardMethod
  23 Friendz -[FriendzAppDelegate showNewMessageWindowWithRecipients:] /Path/To/FriendzAppDelegate.m:226
  22 AppKit -[NSWindowController showWindow:]
  21 AppKit -[NSWindow makeKeyAndOrderFront:]
  20 AppKit -[NSWindow _makeKeyRegardlessOfVisibility]
  19 AppKit -[NSWindow _changeKeyAndMainLimitedOK:]
  18 AppKit -[NSWindow becomeKeyWindow]
  17 AppKit _NXResetCursorState
  16 AppKit +[NSEvent _discardCursorEventsForWindowNumber:criteria:]
  15 HIToolbox FlushSpecificEventsFromQueue
  14 HIToolbox PullEventsFromWindowServer
  13 HIToolbox PullEventsFromWindowServerOnConnection(unsigned int, unsigned char)
  12 HIToolbox ConvertPlatformEventRecordAndPostWithOptions(__CGEvent*, _CGSEventRecord const*, short, unsigned char, unsigned char)
  11 HIToolbox PostEventToQueueInternal
  10 HIToolbox _NotifyEventLoopObservers
   9 HIToolbox KeyEventPostedObserver
   8 HIToolbox TSMProcessRawKeyCode
   7 HIToolbox TSMTranslateKeyEvent
   6 HIToolbox GetDataFromUCHRForEvent
   5 HIToolbox ConvertEventUniCharsToCharCodes
   4 HIToolbox utGetInputSourceScriptInfo
   3 CoreFoundation CFLocaleCreateCanonicalLocaleIdentifierFromScriptManagerCodes
   2 CoreFoundation CFStringCreateWithCStringNoCopy
   1 CoreFoundation __CFStringCreateImmutableFunnel3
   0 CoreFoundation _CFRuntimeCreateInstance

windowControllersNSMutableArray中的applicationDidFinishLaunching alloc / init',并在dealloc方法中发布。

NewMessageWindowController中,我使用以下内容通知应用程序代表该窗口即将关闭,并且不再需要保留控制器:

- (void)windowWillClose:(NSNotification *)notification {
    [AppDelegate windowControllerDidFinish:self];
}

app delegate的方法如下所示:

- (void)windowControllerDidFinish:(NSWindowController *)controller {
    [windowControllers removeObject:controller];
}

之前和之后记录数组是我期望的。在窗口关闭之前,控制器就在那里,当窗口关闭时它被移除。

当我关闭窗户时,仪器会发现泄漏。虽然它是开放的,但一切似乎都很好。值得注意的是,正如预期的那样在NewMessageWindowController中调用dealloc。 Leaks并没有将控制器本身报告为问题,相反,泄漏的对象是NSCFString,它只是源自上面的代码。

构建和分析没有采取任何措施,我很确定我的内存管理在创建/显示窗口控制器/窗口的代码块上很好。

奇怪的是,如果我使用键盘关闭窗口,只会有泄漏。如果我点击红色关闭按钮,仪器不会选择任何东西。

最后,Instruments并不总是表明要负责的代码块。在那些情况下,我的代码中没有一个被引用 - 它似乎都是AppKit。同样,这只有在我使用键盘关闭窗口(cmd-w)时才会出现。

有什么想法吗?

2 个答案:

答案 0 :(得分:0)

在这种情况下,我要做的是,使用Xcode4中的Instruments版本,配置分配工具以记录保留/释放事件。这应该告诉你,对于这个特定的控制器,为什么它的保留计数不会为零。请注意,基于鼠标和基于键盘的关闭可能会执行不同的代码路径。

答案 1 :(得分:0)

这是因为当一个对象被“销毁”时你不能指望dealloc被可靠地调用 - 可能因为某些原因使用键盘不太可能导致立即调用dealloc而不是单击X?