我的代码出现了非常奇怪的错误。事实上,根本没有错误,只是调试器以“程序接收信号:”EXC_BAD_ACCESS“”消息开始。 谁能帮我?我完全糊涂了......谢谢你。
-(NSString *)fullNameForPhone:(NSString *)ph withAlternativeText:(NSString *)text
{
ABAddressBookRef addressBookRef = ABAddressBookCreate();
NSLog(@"create addressBookRef");
NSString *stringToReturn = text;
CFArrayRef allPeopleRef = ABAddressBookCopyArrayOfAllPeople(addressBookRef);
NSLog(@"create allPeopleRef");
CFIndex nPeople = ABAddressBookGetPersonCount(addressBookRef);
int i = 0;
BOOL nameFound = NO;
while ((i < nPeople) && (!nameFound))
{
ABRecordRef recordRef = CFArrayGetValueAtIndex(allPeopleRef, i);
NSLog(@" create recordRef");
CFStringRef allRecordPhonesRef = ABRecordCopyValue(recordRef, kABPersonPhoneProperty);
NSLog(@" create allRecordPhonesRef");
CFIndex nPhones = ABMultiValueGetCount(allRecordPhonesRef);
int currentPhone = 0;
for (currentPhone = 0; currentPhone < nPhones; currentPhone++)
{
CFStringRef currentPhoneNumberRef = ABMultiValueCopyValueAtIndex(allRecordPhonesRef, currentPhone);
NSLog(@" create currentPhoneNumberRef");
NSString *currentCleanPhoneNumber = [self cleanPhoneNumberForString:[NSString stringWithFormat:@"%@", currentPhoneNumberRef]];
if (currentPhoneNumberRef!=NULL)
{
NSLog(@" release currentPhoneNumberRef");
CFRelease(currentPhoneNumberRef);
}
if ([ph isEqualToString:currentCleanPhoneNumber])
{
CFStringRef firstName = ABRecordCopyValue(recordRef, kABPersonFirstNameProperty);
CFStringRef lastName = ABRecordCopyValue(recordRef, kABPersonLastNameProperty);
NSString *fullName = [self fullNameForFirstName:[NSString stringWithFormat:@"%@", firstName]
andLastName:[NSString stringWithFormat:@"%@", lastName]];
if (firstName != NULL)
CFRelease(firstName);
if (lastName != NULL)
CFRelease(lastName);
stringToReturn = fullName;
nameFound = YES;
break;
}
}
CFRelease(allRecordPhonesRef);
NSLog(@" release allRecordPhonesRef");
CFRelease(recordRef);
NSLog(@" release recordRef");
i++;
}
CFRelease(allPeopleRef);
NSLog(@"release allPeopleRef");
CFRelease(addressBookRef);
NSLog(@"release addressBookRef");
return stringToReturn;
}
控制台输出为:
2009-07-31 00:20:05.230 abmodular[21747:20b] create addressBookRef
2009-07-31 00:20:05.231 abmodular[21747:20b] create allPeopleRef
2009-07-31 00:20:05.231 abmodular[21747:20b] create recordRef
2009-07-31 00:20:05.232 abmodular[21747:20b] create allRecordPhonesRef
2009-07-31 00:20:05.232 abmodular[21747:20b] create currentPhoneNumberRef
2009-07-31 00:20:05.232 abmodular[21747:20b] release currentPhoneNumberRef
2009-07-31 00:20:05.232 abmodular[21747:20b] create currentPhoneNumberRef
2009-07-31 00:20:05.233 abmodular[21747:20b] release currentPhoneNumberRef
2009-07-31 00:20:05.233 abmodular[21747:20b] release allRecordPhonesRef
2009-07-31 00:20:05.233 abmodular[21747:20b] release recordRef
2009-07-31 00:20:05.233 abmodular[21747:20b] create recordRef
2009-07-31 00:20:05.234 abmodular[21747:20b] create allRecordPhonesRef
2009-07-31 00:20:05.234 abmodular[21747:20b] create currentPhoneNumberRef
2009-07-31 00:20:05.234 abmodular[21747:20b] release currentPhoneNumberRef
2009-07-31 00:20:05.234 abmodular[21747:20b] release allRecordPhonesRef
2009-07-31 00:20:05.235 abmodular[21747:20b] release recordRef
2009-07-31 00:20:05.235 abmodular[21747:20b] release allPeopleRef
[Session started at 2009-07-31 00:20:05 +0400.]
GNU gdb 6.3.50-20050815 (Apple version gdb-966)
....
Attaching to process 21747.
kill
quit
The Debugger has exited with status 0.(gdb)
按继续输出“EXC_BAD_ACCESS”消息。 Xcode显示,我的代码中最新执行的字符串是CFRelease(addressBookRef);
答案 0 :(得分:15)
我有同样的问题做类似的事情,经过进一步的研究,我发现我已经过度释放了。根据{{3}}:
如果您创建或复制Core 基础对象,你必须 随后在你发布时释放它 完成它。
我读到这意味着你不应该释放带有单词Get的函数。如果这样做,稍后当真正的所有者试图释放它时将导致问题。所以,在这种情况下你这样做:
ABRecordRef recordRef = CFArrayGetValueAtIndex(allPeopleRef, i);
以后:
CFRelease(recordRef);
你正在发布一些不应该被释放的东西。很久以后你做的事情:
CFRelease(allPeopleRef);
该阵列将尝试释放其所有记录,而不知道您已经释放了其中的一些记录。结果就是你的错误。通过注释掉那条线,你可能会让错误消失,但我担心你造成了内存泄漏。
我建议您不要在获取方法指针上调用CFRelease
,并在创建或复制方法指针上调用它(这条规则可能有例外,但到目前为止它对我有用。)
答案 1 :(得分:-2)
allPeopleRef和addressBookRef指向相同的对象?副本可能很浅。 ABAddressBookCopyArrayOfAllPeople做了什么?