我正在运行后台线程以从Web服务获取数据。
为了获得创建URL所需的设置,我正在使用此方法:
+(Settings *)getSettings
{
Settings *settings = [[Settings alloc] init];
NSString *path = [NSString stringWithFormat:@"Documents/Settings"];
NSString *settingsPath = [NSHomeDirectory() stringByAppendingPathComponent:path];
NSFileManager *fileMgr = [NSFileManager defaultManager];
if ([fileMgr fileExistsAtPath:settingsPath]) {
NSDictionary *settingsDictionary = [NSDictionary dictionaryWithContentsOfFile:settingsPath];
settings.username = [settingsDictionary objectForKey:@"username"];
settings.module = [settingsDictionary objectForKey:@"module"];
settings.websiteUrl = [settingsDictionary objectForKey:@"websiteUrl"];
}else
{
settings.username = @"";
settings.module = @"";
settings.websiteUrl = @"";
}
NSLog(@"Settings = u:%@ m:%@ w:%@",settings.username,settings.module,settings.websiteUrl);
return settings;
}
这在调试器中工作正常,但是当我在设备上运行它时,我在NSLog或试图使用带有警告的数据的任何类上得到BAD_EXCESS:
警告:尝试使用不在框架中的块创建USE_BLOCK_IN_FRAME变量。
如果我单步执行代码,则该代码在设备上可以正常工作。
有任何想法吗?
编辑:
回溯-
#0 0x37a97eda in objc_retain ()
#1 0x37aa4be4 in objc_retainAutoreleasedReturnValue ()
#2 0x0000f84a in +[SettingData getSettings] (self=0x175e4, _cmd=0x11af9) at /Users/Jono/Dropbox/Uni/iOS5/Feedback At Tees/Feedback At Tees/SettingData.m:49
#3 0x0000fa86 in -[DataManager init] (self=0x3509e0, _cmd=0x30fd9f96) at /Users/Jono/Dropbox/Uni/iOS5/Feedback At Tees/Feedback At Tees/DataManager.m:19
#4 0x00003b46 in __35-[MasterViewController viewDidLoad]_block_invoke_0 (.block_descriptor=0x185f40) at /Users/Jono/Dropbox/Uni/iOS5/Feedback At Tees/Feedback At Tees/MasterViewController.m:70
#5 0x30b08d54 in _dispatch_call_block_and_release ()
#6 0x30b0b896 in _dispatch_worker_thread2 ()
#7 0x37a0e1ce in _pthread_wqthread ()
#8 0x37a0e0a4 in start_wqthread ()
我尝试将以下内容添加到设置的init方法中,如果我不这样做,如果我不这样做,它会在第二个NSLog上崩溃,并显示以下错误:消息已发送到已释放实例0x160d60
-(id)init
{
if ((self=[super init]))
{
NSString *path = [NSString stringWithFormat:@"Documents/Settings"];
NSString *settingsPath = [NSHomeDirectory() stringByAppendingPathComponent:path];
NSFileManager *fileMgr = [NSFileManager defaultManager];
if ([fileMgr fileExistsAtPath:settingsPath])
{
NSDictionary *settingsDictionary = [NSDictionary dictionaryWithContentsOfFile:settingsPath];
self.username = [settingsDictionary objectForKey:@"username"];
self.module = [settingsDictionary objectForKey:@"module"];
self.websiteUrl = [settingsDictionary objectForKey:@"websiteUrl"];
NSLog(@"Settings = u:%@ m:%@ w:%@",self.username,self.module,self.websiteUrl);
}
else
{
self.username = @"";
self.module = @"";
self.websiteUrl = @"";
}
}
NSLog(@"Settings = u:%@ m:%@ w:%@",self.username,self.module,self.websiteUrl);
return self;
}
屏幕截图:
http://postimage.org/image/4vw6uq7pp/
最佳答案
首先,该方法应该是retrieveSettings
或downloadSettings
,或者只是settings
,但不应该是getSettings
,因为get
作为前缀对于方法的论证(通过引用传递)具有非常特殊的含义。
其次,该代码没有任何明显错误(除了settings
的泄漏,但这不会导致这种崩溃)。 (我看到启用了ARC)。
最后,如果发生崩溃,则发布backtrace 。您看到的错误消息(“USE_BLOCK_IN_FRAME”)表示您发布的代码与崩溃无关。
奇怪的崩溃。好的-警告来自调试器,与实际错误无关。
范围边界通常是编译器将发出release
和/或排出自动释放池的位置。
这就引出了下一个问题;如何
如果我单步执行代码,则该代码在设备上可以正常工作。
好的-因此,当您全速运行时它不起作用,而在逐步运行时它可以起作用。闻起来像并发崩溃。您可以发布实际崩溃的屏幕截图吗?即您确定特定的回溯是崩溃的回溯,而不是调试器在崩溃时碰巧的位置吗?
(是的,稍稍抓住稻草-崩溃与所显示的证据无关。我喜欢拼图。)
好-现在我们到了。有点。成功在if范围内记录相同内容后,它将在if范围之外崩溃。
如何定义username
,module
和websiteURL
?它们是strong
还是最合适的copy
@property
声明?
关于iphone - iOS5 BAD_EXCESS仅在设备上,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/10194923/