gpt4 book ai didi

iphone - NSMutableArray 与 NSMutableDictionary 的一个奇怪问题

转载 作者:行者123 更新时间:2023-12-03 20:28:03 25 4
gpt4 key购买 nike

我有一个 XML 解析器,它可以将 XML 文件(作为项目资源附加)中的所有信息解析到集合中。第一次我通过 NSMutableArray 完成它,然后将其转换为 NSMutableDictionary ,一切正常。之后我意识到没有必要有一个 NSMutableArray;我可以直接将 XML 解析为 NSMutableDictionary。我做到了(这只是 1 个属性和我将对象添加到集合中的方式的问题),但问题是我现在遇到了泄漏。非常奇怪的泄漏。它可能发生,又不可能发生。每次发生这种情况时,它都会说与哈希和键/值问题有关。部分代码:

- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName attributes:(NSDictionary *)attributeDict{
self.currentElement = elementName;
// category
if(self.targetBranch==XML_KEY_CATEGORY && [elementName isEqualToString:XML_KEY_CATEGORY]) {
self.categoryItem = [[CategoryItem alloc] initWithId:[attributeDict objectForKey:@"id"] andParentId:[attributeDict objectForKey:@"parentId"]];
//NSLog(@"didStartElement with id %@",self.categoryItem.categoryId);
}
...
}

- (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string{
...
// categories
if (self.targetBranch==XML_KEY_CATEGORY && [self.currentElement isEqualToString:XML_KEY_CATEGORY]) {
//NSLog(@"adding category to array");
if (self.categoryItem.categoryName)
self.categoryItem.categoryName = [NSString stringWithFormat:@"%@%@",self.categoryItem.categoryName,string]; //[string copy];
else
self.categoryItem.categoryName = string; //[string copy];
...
}


- (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI qualifiedName:(NSString *)qName{
//NSLog(@"ended element: %@", elementName);

// category
if (self.targetBranch==XML_KEY_CATEGORY && [elementName isEqualToString:XML_KEY_CATEGORY]) {
//NSLog(@"adding category to array");
//[xmlParsedInfo addObject:self.categoryItem];
// if (self.categoryItem.categoryId && self.categoryItem.categoryId && self.categoryItem.categoryId!=NULL
// && ![xmlParsedData objectForKey:self.categoryItem.categoryId]){
// //NSLog(@"adding dictionary element %@",self.categoryItem.categoryId);
// [xmlParsedData setObject:self.categoryItem forKey:self.categoryItem.categoryId];
[xmlParsedInfo addObject:self.categoryItem];

}
...
}

-(NSDictionary*) getCategoriesForParentId:(NSString*)parentId_ fromUrl:(NSString *)strUrl{
NSLog(@"getCategoriesForParentId: %@",parentId_);
targetBranch = XML_KEY_CATEGORY;
parentId = parentId_;
xmlParsedInfo = [[NSMutableArray alloc] init];
// xmlParsedData = [[NSMutableDictionary alloc] init];
[self parseXMLFileAtURL:strUrl];
NSLog(@"category for parent id count is %i",xmlParsedInfo.count);
//return xmlParsedData;
NSDictionary *result = [self convertCategoryItemArrayToDictionary:xmlParsedInfo];
[xmlParsedInfo release];
return result; //[self convertCategoryItemArrayToDictionary:xmlParsedInfo];
}

- (NSDictionary *) convertCategoryItemArrayToDictionary:(NSMutableArray *)array
{
NSMutableDictionary *mutableDictionary = [[NSMutableDictionary alloc] init];
for (CategoryItem *catItem in array) {
[mutableDictionary setObject:catItem forKey:catItem.categoryId];
}
return (NSDictionary *)[mutableDictionary autorelease];
}

正如我所说,XML 解析器工作正常,目前还没有出现任何问题。但您可以看到与字典相关的注释部分。在当前状态下,它运行良好,但如果注释数组相关代码和取消注释字典相关 - 它将泄漏。大多数时候我会遇到 2 次泄漏,例如:

#   Category    Event Type  Timestamp   RefCt   Address Size    Responsible Library Responsible Caller
0 CFBasicHash (key-store) Malloc 1928615936 1 0x9019600 512 MyApp -[XMLParser parser:didEndElement:namespaceURI:qualifiedName:]
# Category Event Type Timestamp RefCt Address Size Responsible Library Responsible Caller
0 CFBasicHash (value-store) Malloc 1928608768 1 0x900c600 512 MyApp -[XMLParser parser:didEndElement:namespaceURI:qualifiedName:]

这些泄漏直接导致以下代码行:

        [xmlParsedData setObject:self.categoryItem forKey:self.categoryItem.categoryId];

那么这里的字典有什么问题呢?正如我所看到的,它的哈希/ key 相关以及它的 NSMutableDictionary 内部问题。

我仅使用模拟器尝试代码。所以我不能说问题是否仍然存在于真实设备上。也许它只是与模拟器相关的错误(例如键盘泄漏)。
我正在解析的 XML 是静态的,它只是项目中的一个文件。此 XML 文件中有 71 个类别。因此,每个运行的应用程序都会使用相同的数据执行相同的操作。这就是为什么我说这个问题很奇怪。第一次使用泄漏监视器运行时,没有任何泄漏。在第二次尝试时,它可能是 1 或 2。接下来的所有运行都会不断显示 2 个(如上所示)泄漏(+1 导致字典本身,但这是由于其键或值泄漏,它只是链的末端) .

CategoryItem.h
@interface CategoryItem : NSObject {
NSString * categoryName;
NSString * categoryId;
NSString * parentId;
}
@property (nonatomic,retain) NSString * categoryName;
@property (nonatomic,retain) NSString * categoryId;
@property (nonatomic,retain) NSString * parentId;

- (id)init;
- (id)initWithId:(NSString *)id_ andParentId:(NSString *)parentId_;
- (id)initWithName:(NSString *)name_ andId:(NSString *)id_ andParentId:(NSString *)parentId_;
- (BOOL)isEqualToItem:(CategoryItem *)anItem;
- (NSUInteger)hash;

@end

CategoryItem.m

#import "CategoryItem.h"

@implementation CategoryItem

@synthesize categoryName;
@synthesize categoryId;
@synthesize parentId;

-(id)init{
self = [super init];
return self;
}

-(id)initWithName:(NSString *)name_ andId:(NSString *)id_ andParentId:(NSString *)parentId_{
self = [super init];
if(self){
self.categoryName = name_;
self.categoryId = id_;
self.parentId = parentId_;
}
return self;
}

-(id)initWithId:(NSString *)id_ andParentId:(NSString *)parentId_{
self = [super init];
if(self){
self.categoryId = id_;
self.parentId = parentId_;
}
return self;
}

- (void)dealloc {
[categoryName release];
[categoryId release];
[parentId release];
[super dealloc];
}

- (BOOL)isEqual:(id)other {
if (other == self)
return YES;
if (!other || ![other isKindOfClass:[self class]])
return NO;
return [self isEqualToItem:other];
}

- (BOOL)isEqualToItem:(CategoryItem *)anItem {
if (self == anItem)
return YES;
if (![(id) self.categoryName isEqual:anItem.categoryName])
return NO;
if (![(id) self.categoryId isEqual:anItem.categoryId])
return NO;
if (![(id) self.parentId isEqual:anItem.parentId])
return NO;
return YES;
}

- (NSUInteger)hash {
NSString *string4Hash = [NSString stringWithFormat:@"%@%@%@",self.categoryName,self.categoryId,self.parentId];
NSUInteger hash = [string4Hash hash];
return hash;
}

@end

添加:我找到的解决方案是(对我来说很愚蠢)在我的 -(NSDictionary*) getCategoriesForParentId:

返回此:

    NSDictionary *result;

result = (NSDictionary *)[xmlParsedData copy];
[xmlParsedData release];

return result;

!或者!

    return [(NSDictionary *)xmlParsedData autorelease];

}

所以重点是复制 NSDictionaryMutable...我只是不明白为什么它有帮助?!不是保留而是复制!调用 getCategoriesForParentId() 的方法需要 NSDictionary (而不是可变的)。它通过 alloc+init 创建 XMLParser,并在获取字典后释放创建的 XMLParser,该 XMLParser 具有 dealloc

- (void)dealloc {
[currentElement release];
[targetBranch release];

xmlParsedInfo = nil;
xmlParsedData = nil;
...

我还尝试返回类似 (NSDictionary *)[xmlParsedData keep] 或 (NSDictionary *)xmlParsedData 的内容,而不需要额外的 NSDictionary *结果 - 但存在泄漏。

看起来问题是我没有使用 xmlParsedData 作为属性+@synthesize。所以现在我将其设为一个属性(保留),并在 dealloc 中将其现在 [xmlParsedData 发布] 中,并且我从 getCategoriesForParentId 返回 [xmlParsedData 保留]。还是漏了。

最佳答案

您的代码似乎没有使用 ARC,但请在问题中指定它。

每次进入 didStartElementif 分支并将新元素分配给 self.categoryItem 时,都会发生内存泄漏。您正在分配一个新对象,但是当您将其分配给属性时,它将再对其发出一个 retain 。下次当您分配下一个对象时,该属性将发出一个release,但引用计数器仍将比您alloc之后所需的多一个。

为了使其正常工作,请在创建时发出自动释放或在将其分配给属性后发出释放:

self.categoryItem = [[[CategoryItem alloc] 
initWithId:[attributeDict objectForKey:@"id"]
andParentId:[attributeDict objectForKey:@"parentId"]] autorelease];

关于iphone - NSMutableArray 与 NSMutableDictionary 的一个奇怪问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10242006/

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