我一直在努力解决以下泄漏问题。我通过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
windowControllers
是NSMutableArray
中的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)时才会出现。
有什么想法吗?
答案 0 :(得分:0)
在这种情况下,我要做的是,使用Xcode4中的Instruments版本,配置分配工具以记录保留/释放事件。这应该告诉你,对于这个特定的控制器,为什么它的保留计数不会为零。请注意,基于鼠标和基于键盘的关闭可能会执行不同的代码路径。
答案 1 :(得分:0)
这是因为当一个对象被“销毁”时你不能指望dealloc被可靠地调用 - 可能因为某些原因使用键盘不太可能导致立即调用dealloc而不是单击X?