iPhone SDK : EXC_BAD_ACCESS with CFRelease for ABAddressBookRef

标签 iphone sdk addressbook

我的代码有一个非常奇怪的错误。事实上,根本没有错误,只是调试器以“程序收到信号:“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);

最佳答案

我在做类似的事情时遇到了同样的问题,经过进一步的研究,我发现我过度释放了。根据Core Foundation docs :

If you create or copy a Core Foundation object, you must subsequently release it when you’re finished with it.

我认为这意味着您不应该发布带​​有 Get 一词的函数。如果这样做,当真正的所有者尝试释放它时,将会导致问题。因此,在这种情况下,当您这样做时:

ABRecordRef recordRef = CFArrayGetValueAtIndex(allPeopleRef, i);

及以后:

CFRelease(recordRef);

您正在发布一些不应该发布的内容。很久以后,当你这样做时:

CFRelease(allPeopleRef);

数组将尝试释放其所有记录,但不知道您已经释放了其中一些记录。结果是你的错误。通过注释掉该行,您可能已经使错误消失,但我担心您已经造成了内存泄漏。

我建议您不要在Get方法指针上调用CFRelease,而是在CreateCopy上调用它方法指针(此规则可能有异常(exception),但到目前为止它对我有用)。

关于iPhone SDK : EXC_BAD_ACCESS with CFRelease for ABAddressBookRef,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1209130/

相关文章:

ios - 如何显示带有事件指示器的警报 View ?

android - Titanium 3.1.0 - 模拟器进程退出,代码为 1 - 无法编译 APK

cocoa - kAB*** 属性列表?

ios - 字符串中奇怪的不可见字符导致 NSURL 失败

iphone - 使用 Google Analytics iOS SDK 跟踪应用内购买

iPhone SDK : NSUserDefaults or NSDictionary?

iphone - 在 Xcode 之外为 iPhone 进行开发

java - 尝试使用 android SDK 运行 "Panic"程序时出现 "Hello world"错误

android - 创建虚拟设备出现错误 "The skin directory does not point to a valid skin"

iphone - 将联系人列表保存到 NSUserDefaults