2

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パーサーは正常に機能し、まだ問題はありません。しかし、あなたは辞書に関連しているコメントされたセクションを見ることができます。現在の状態では正常に動作しますが、コメント配列に関連するコードとUNcomment辞書に関連する場合はリークします。ほとんどの場合、次のような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];

では、ここでの辞書の問題は何ですか?私が見ることができるように、そのハッシュ/キー関連とそのNSMutableDictionary内部の問題。

エミュレーターのみでコードを試しています。したがって、問題が実際のデバイスで引き続き発生するかどうかはわかりません。たぶん、エミュレータに関連するバグ(キーボードリークなど)です。
私が解析しているXMLは静的であり、プロジェクト内の単なるファイルです。このXMLファイルには71のカテゴリがあります。したがって、すべての実行アプリは同じデータで同じことを行います。だから私は問題が奇妙だと言っているのです。リークモニターを使用した最初の実行では、リークはありません。2回目の試行では、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(Mutableではない)を予期します。alloc + initを介してXMLParserを作成し、辞書を取得した後、deallocを持つ作成されたXMLParserをリリースします。

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

    xmlParsedInfo = nil;
    xmlParsedData = nil; 
...

また、(NSDictionary *)[xmlParsedDataretain]や(NSDictionary *)xmlParsedDataのようなものを追加のNSDictionary * resultなしで返そうとしましたが、リークがありました。

問題は、xmlParsedDataをproperty +@synthesizeとして使用していないことのようです。だから今私はそれをプロパティ(保持)にし、deallocでその今[xmlParsedDataリリース]を取得し、getCategoriesForParentIdから[xmlParsedData保持]を返します。それはまだ漏れています。

4

2 に答える 2

5

コードは ARC を使用していないようですが、質問で指定してください。

ifブランチに入っdidStartElementて新しい要素を に割り当てるたびに、メモリがリークしますself.categoryItem。新しいオブジェクトを割り当てていますが、それをプロパティに割り当てると、もう 1 つ発行retainされます。次に次のオブジェクトを割り当てると、プロパティはそれを発行しますreleaseが、参照カウンターはalloc.

うまく機能させるには、作成時に autorelease を発行するか、プロパティに割り当てた後に release を発行します。

self.categoryItem = [[[CategoryItem alloc] 
         initWithId:[attributeDict objectForKey:@"id"] 
        andParentId:[attributeDict objectForKey:@"parentId"]] autorelease];
于 2012-04-20T08:10:25.567 に答える
1

これが問題かどうかはわかりませんが...categoryItemクラスを投稿できますか?または...ハッシュメソッドを実装しましたか?

于 2012-04-20T07:51:01.420 に答える