gpt4 book ai didi

ios - 解决 Xcode 中的内存泄漏

转载 作者:行者123 更新时间:2023-11-29 13:08:19 24 4
gpt4 key购买 nike

我正在尝试使用地址簿在联系人 APP 上做一个程序,它工作正常,但是当我分析有几个内存泄漏时,我已经设法将内存泄漏降至最低,现在我只剩下 2 个主要内存泄漏警告,一个在我的地址簿重新加载功能中,我包含了评论以显示我尝试过的所有事情

-(void)reloadAddressBook
{
//if(self.addressBook)
//CFRelease(self.addressBook);
self.addressBook = (__bridge ABAddressBookRef) CFBridgingRelease(ABAddressBookCreate());

if(ABAddressBookHasUnsavedChanges(self.addressBook))
{

ABAddressBookSave(self.addressBook,NULL);


}


//if(self.contactAdd)
//CFRelease(self.contactAdd);
self.contactAdd= ABAddressBookCopyArrayOfAllPeople(self.addressBook);


**//Memory warning here and says: call to function ABAddressBookCopyArrayOfAllPeople returns a core foundation object with a +1 retain count**
//self.contactAdd= (__bridge ABAddressBookRef) CFBridgingRelease(ABAddressBookCopyArrayOfAllPeople(self.addressBook));
**// If I use this format my memory leak issue solves here but I get error in my program**
}

- (void)viewDidLoad
{**//Memory warning here and says :object leaked :allocated object is not retained lated in this execution path and has retain count +1**
[super viewDidLoad];

self.contactSearchBar.delegate=self;
self.contactTableView.delegate=self;
self.contactTableView.dataSource=self;

UIBarButtonItem *addContactButton=[[UIBarButtonItem alloc]initWithTitle:@"Add" style:UIBarButtonItemStyleBordered target:self action:@selector(newContact:)];
self.navigationItem.rightBarButtonItem=addContactButton;
self.navigationItem.title=@"My Contacts";


}

另一个内存泄漏是在这个搜索栏函数中

-(void)searchBar:(UISearchBar *)searchBar textDidChange:(NSString *)searchText
{

if(searchText.length==0)
{
isFiltered=NO;

}
else
{
isFiltered=YES;

int j=0,i=0;
self.filteredData= CFArrayCreateMutable(kCFAllocatorDefault, 0,&kCFTypeArrayCallBacks);**// Memory warning here and says: call to function CFArrayCreateMutable returns a core foundation object with a +1 retain count**

for(i=0;i<CFArrayGetCount(self.contactAdd);i++)**//Memory warning here and says :object leaked :allocated object is not retained lated in this execution path and has retain count +1**
{
self.person=CFArrayGetValueAtIndex(self.contactAdd,i);
NSString *str=[[NSString stringWithFormat:@"%@", (__bridge_transfer NSString *)ABRecordCopyValue(self.person, kABPersonFirstNameProperty)] stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]];
NSRange contactRange= [str rangeOfString: searchText options:NSCaseInsensitiveSearch];

NSLog(@"i=%d, j=%d",i,j);

if(contactRange.location!=NSNotFound)
{
CFArrayInsertValueAtIndex(self.filteredData,j++,self.person);
CFArrayGetValueAtIndex(self.filteredData,j-1);
}

}
//CFRelease(self.contactAdd);
//CFRelease(self.filteredData);
}

for 循环语句显示内存泄漏,它说:

最佳答案

就像我在 my answer on your other question 中提到的那样, in Core Foundation, Create and Copy functions return an ownership reference.

持有对象的属性通常也拥有引用,除非您以其他方式声明它们(通常使用 weak)。这意味着,在这样的声明中:

self.filteredData= CFArrayCreateMutable(…);

您现在两次拥有该对象:一次是因为您创建了它,一次是因为您的属性(property)保留了它。

对于每个属性、ivar 或其他强引用(例如局部变量),您通常应该只拥有每个对象一次。任何额外的所有权(例如来自 Create 和 Copy 函数的)都是您需要清理的东西,这就是为什么最好在 CF 领域做的越少越好:ARC 会为您清理东西,但它不会触及 CF除非你告诉它。

说到属性和局部变量,您不需要将所有内容都设为属性。搜索代码中的 person 应该是该方法中的局部变量,因为它是该搜索状态的一部分,而不是您的对象需要无限期保留的东西。您已经将 str 作为局部变量,所以我不确定您为什么要为 person 使用属性。

您告诉 ARC 通过桥梁 Actor 与 CF 世界互动:

  • __bridge_transfer(或 CFBridgingRelease)告诉 ARC“当我用完它时为我清理这个所有权”。
  • __bridge_retained(或 CFBridgingRetain)告诉 ARC“我要把这个对象扔进 CF-land;除非我另有说明,否则不要放弃它”(然后您必须在某个时候通过传输或立即 CFRelease 执行此操作)。

再说一次,您通常希望从 CF 世界中取出一些东西,让 ARC 尽可能多地处理它们。

因此,我建议将 self.filteredData 设为 NSMutableArray,而不是 CFMutableArray。使用 Foundation 创建阵列意味着它已经在 ARC 的控制之下,无需桥接它。

对于 self.contactAdd,您可以在检索它时将其桥接为 NSArray,然后从那时起将其视为 NSArray。这包括使用快速枚举而不是索引来遍历它,或者更好的是,使用谓词搜索而不是自己遍历它。 (我不指望内置的比较谓词能够与 ABPersons 一起工作,但这就是 predicateWithBlock: 的用途。)

搜索代码的基于谓词的版本看起来像这样(完全未经测试):

self.filteredData = [self.contactAdd filteredArrayUsingPredicate:[NSPredicate predicateWithBlock:^BOOL(id evaluatedObject, NSDictionary *bindings){
ABPersonRef person = (__bridge ABPersonRef)evaluatedObject;
NSString *firstName = [[NSString stringWithFormat:@"%@", (__bridge_transfer NSString *)ABRecordCopyValue(self.person, kABPersonFirstNameProperty)] stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceCharacterSet]];

NSRange contactRange = [str rangeOfString:searchText options:NSCaseInsensitiveSearch];
return (contactRange.location != NSNotFound);
}];

(该 block 和基于它的谓词返回每个对象是否匹配。filteredArrayUsingPredicate: 创建一个数组,其中包含谓词评估为 true 的每个对象>.)

还有一件事:您是否考虑过使用 ABPeoplePickerNavigationController?

关于ios - 解决 Xcode 中的内存泄漏,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18013172/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com