3

更新 | 我已经使用面板​​上传了一个示例项目并在此处崩溃:http ://w3style.co.uk/~d11wtq/BlocksCrash.tar.gz (我知道“选择...”按钮什么也没做,我没有尚未实施)。

更新 2 | 刚刚发现我什至不需要调用任何东西newFilePanel来导致崩溃,我只需要在语句中使用它。

这也会导致崩溃:

[newFilePanel beginSheetModalForWindow:[windowController window] completionHandler:^(NSInteger result) {
    newFilePanel; // Do nothing, just use the variable in an expression
}];

似乎转储到控制台的最后一件事有时是:“无法反汇编 dyld_stub_objc_msgSend_stret。”,有时是:“无法访问地址 0xa 的内存”。

我创建了自己的工作表(一个 NSPanel 子类),它试图提供类似于 NSOpenPanel/NSSavePanel 的 API,因为它将自己呈现为工作表并在完成时调用一个块。

这是界面:

//
//  EDNewFilePanel.h
//  MojiBaker
//
//  Created by Chris Corbyn on 29/12/10.
//  Copyright 2010 Chris Corbyn. All rights reserved.
//

#import <Cocoa/Cocoa.h>

@class EDNewFilePanel;

@interface EDNewFilePanel : NSPanel <NSTextFieldDelegate> {
    BOOL allowsRelativePaths;

    NSTextField *filenameInput;

    NSButton *relativePathSwitch;

    NSTextField *localPathLabel;
    NSTextField *localPathInput;
    NSButton *chooseButton;

    NSButton *createButton;
    NSButton *cancelButton;
}

@property (nonatomic) BOOL allowsRelativePaths;

+(EDNewFilePanel *)newFilePanel;

-(void)beginSheetModalForWindow:(NSWindow *)aWindow completionHandler:(void (^)(NSInteger result))handler;
-(void)setFileName:(NSString *)fileName;
-(NSString *)fileName;
-(void)setLocalPath:(NSString *)localPath;
-(NSString *)localPath;
-(BOOL)isRelative;

@end

以及实现里面的关键方法:

-(void)beginSheetModalForWindow:(NSWindow *)aWindow completionHandler:(void (^)(NSInteger result))handler {
    [NSApp beginSheet:self
       modalForWindow:aWindow
        modalDelegate:self
       didEndSelector:@selector(sheetDidEnd:returnCode:contextInfo:)
          contextInfo:(void *)[handler retain]];
}

-(void)dismissSheet:(id)sender {
    [NSApp endSheet:self returnCode:([sender tag] == 1) ? NSOKButton : NSCancelButton];
}

-(void)sheetDidEnd:(NSWindow *)aSheet returnCode:(NSInteger)result contextInfo:(void *)contextInfo {
    ((void (^)(NSUInteger result))contextInfo)(result);
    [self orderOut:self];
    [(void (^)(NSUInteger result))contextInfo release];
}

如果我的块只是一个空主体的无操作,这一切都有效。当工作表被解除时,我的块被调用。

EDNewFilePanel *newFilePanel = [EDNewFilePanel newFilePanel];
[newFilePanel setAllowsRelativePaths:[self hasSelectedItems]];
[newFilePanel setLocalPath:@"~/"];
[newFilePanel beginSheetModalForWindow:[windowController window] completionHandler:^(NSInteger result) {
    NSLog(@"I got invoked!");
}];

但是,当我尝试从块内部访问面板时,我会因 EXC_BAD_ACCESS 而崩溃。例如,这会崩溃:

EDNewFilePanel *newFilePanel = [EDNewFilePanel newFilePanel];
[newFilePanel setAllowsRelativePaths:[self hasSelectedItems]];
[newFilePanel setLocalPath:@"~/"];
[newFilePanel beginSheetModalForWindow:[windowController window] completionHandler:^(NSInteger result) {
    NSLog(@"I got invoked and the panel is %@!", newFilePanel);
}];

从调试器中不清楚原因是什么。堆栈上的第一项(零 0)只是说“??” 并且没有列出任何内容。

-endSheet:returnCode:堆栈中的下一项(1 和 2)分别是对和的调用-dismissSheet:。查看调试器中的变量,似乎没有任何问题/超出范围。

我原以为面板可能已经发布(因为它是自动发布的),但即使-retain在创建它后立即调用它也无济于事。

我执行这个错误吗?

4

2 回答 2

12

当该对象不是实例变量时,一个方法中retain的参数和另一个方法中的参数对您来说有点奇怪。release

我建议将completionHandler你的beginSheet东西作为实例变量。无论如何,您不能一次多次显示工作表,这样会更干净。

此外,您EXC_BAD_ACCESS很可能来自您方法[handler retain]中的调用。beginSheet:您可能正在使用类似(为简洁起见)调用此方法:

[myObject doThingWithCompletionHandler:^{ NSLog(@"done!"); }];

如果是这种情况,您必须 -copy阻止而不是保留它。如上所述,该块存在于堆栈中。但是,如果该堆栈帧从执行堆栈中弹出,那么该块就消失了。 poof 以后任何访问该块的尝试都将导致崩溃,因为您正在尝试执行不再存在并已被垃圾替换的代码。因此,您必须copy在块上调用以将其移动到堆中,它可以在堆中存活超过创建它的堆栈帧的生命周期。

于 2010-12-30T06:35:40.020 回答
-1

尝试使用修饰符定义您的 EDNewFilePanel __block

__block EDNewFilePanel *newFilePanel = [EDNewFilePanel newFilePanel];

这应该在调用块时保留对象,这可能是在 Panel 对象被释放之后。作为一个不相关的副作用,这也会使其在块范围内可变。

于 2010-12-30T06:24:51.953 回答