0

我是 xcode / cocoa 甚至是 Objective-c 的新手,因此我的问题可能很愚蠢。我正在尝试编写一个程序来散列文件夹中的文件。我环顾四周,找到了一种通过 NSData 对象加载文件并使用 CC_SHA512 散列它的字节的方法。

如果我尝试对更多文件进行哈希处理,我注意到我的内存已用完。使用 Run -> Performance Tools 我可以查明我的问题。我创建的所有 NSData 对象仍在内存中。我尝试使用 release/dealloc 自动释放和手动释放。没有任何工作。

我的编译器设置是标准的,但有一个例外,我选择 Objective-C 垃圾收集 = 必需。

也许有人可以告诉我我做错了什么。

这是代码:

-(FileHash*) hashFileByName :(NSString*) filePath{

    //NSData* inputData = [inputStr dataUsingEncoding:NSUTF8StringEncoding];
    NSLog(filePath);
    NSData* inputData = [[NSData dataWithContentsOfFile:filePath] autorelease];
    unsigned char outputData[CC_SHA512_DIGEST_LENGTH];
    CC_SHA512([inputData bytes], [inputData length], outputData);


    NSMutableString* hashStr = [NSMutableString string];
    int i = 0;
    for (i = 0; i < CC_SHA512_DIGEST_LENGTH; ++i)
        [hashStr appendFormat:@"%02x", outputData[i]];


    //NSLog(@"%@ hash : %@",filePath,hashStr);

    FileHash *hash = [[[FileHash alloc]init]autorelease];
    [hash setFileHash:hashStr];
    [hash setFilePath:filePath];
    [inputdata release];
    [inputdata dealloc];
    return hash;    
}

-(NSMutableArray*) hashFilesInDirectory:(NSString*) pathToDirectory:(Boolean) recursive : (IBOutlet id) Status : (Boolean*) BreakOperation{

    NSGarbageCollector *collect = [NSGarbageCollector defaultCollector];

    NSMutableArray *files;
        files = [[self listFilesOnlyRecursive:pathToDirectory] autorelease];

    NSMutableArray *hashes = [[[NSMutableArray alloc]init]autorelease];

    for (NSString *file in files) {

        [hashes addObject: [self hashFileByName:file]]; 
        [collect collectExhaustively];
    }


    return hashes;
}

-(NSMutableArray*) listFilesOnlyRecursive : (NSString*) startDir {

    NSMutableArray *filelist = [[[NSMutableArray alloc] init]autorelease];

    //Inhalt eines Verzeichnisses auflisten (unterverzeichnisse werden ignoriert
    NSFileManager *manager = [[NSFileManager defaultManager]autorelease];

    NSDirectoryEnumerator *enumerator = [manager enumeratorAtPath:startDir];
    int count = 0;
    id file;
    while (file = [enumerator nextObject])
    {

        //      file = [[[[startDir stringByAppendingString:@"/"]autorelease] stringByAppendingString:file] autorelease
        //              ];
        file = [NSString stringWithFormat:@"%@/%@",startDir,file];
        BOOL isDirectory=NO;
        [[NSFileManager defaultManager] fileExistsAtPath:file isDirectory:&isDirectory];
        if (!isDirectory){
            [filelist addObject:file];
            //printf("\n:%s:\n",[file UTF8String]);
            count++;
        }


    }
    NSLog(@"Es waren %i files",count);
    return filelist;
}

这一切都始于

int main(int argc, char *argv[])
{
    NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
    //return NSApplicationMain(argc,  (const char **) argv);
    MemoryLeakTest *test = [[[MemoryLeakTest alloc]init]autorelease];
    [test hashFilesInDirectory:@"/huge directory/" :YES :nil :nil];
    [pool drain];
    [pool release];
    [pool dealloc];

}

也许有人有一个想法。

比你提前:) Nubus

4

2 回答 2

2

有几件事:

NSData* inputData = [[NSData dataWithContentsOfFile:filePath] autorelease];

如果您想在当前方法执行之后保留它,则必须保留它(这有点过于简单,但对于您的示例来说是合理的)。由于dataWithContentsOfFile:不包含 alloc、copy 或 new,除非您明确保留它,否则您不负责释放它。鉴于您只在函数中本地使用它,您不需要保留它。因此,只需使用以下内容,不要在其上调用 release/autorelease 或 dealloc:

NSData* inputData = [NSData dataWithContentsOfFile:filePath];

此外,您永远不会手动释放事物。只需根据需要释放/自动释放它们。dealloc 将根据需要被调用。

[inputData dealloc]; // don't do this explicitly

您肯定需要阅读Cocoa 内存管理文档。它会清除很多。

于 2009-11-14T19:12:07.247 回答
0

在阅读了内存管理文档后,我知道的和以前一样多。所以我开始尝试和错误。我尝试了诸如释放 NSData 对象之类的方法,直到保留计数为 0 .. and and and 。比我找到了一个可行的解决方案。

我必须自己初始化 NSData 对象,并将其设置为自动释放(我无法自己释放它,因为在我调用 init 后,retaincound 为 2 并且尝试释放它 2 次会导致崩溃

blabla这里是我的解决方案:

-(FileHash*) hashFileByName :(NSString*) filePath{

    NSAutoreleasePool *innerpool = [[NSAutoreleasePool alloc]init];

    //NSData* inputData = [inputStr dataUsingEncoding:NSUTF8StringEncoding];
    NSData* inputData = [[[NSData alloc]initWithContentsOfFile:filePath] autorelease];
    unsigned char outputData[CC_SHA512_DIGEST_LENGTH];
    CC_SHA512([inputData bytes], [inputData length], outputData);


    NSMutableString* hashStr = [NSMutableString string];
    int i = 0;
    for (i = 0; i < CC_SHA512_DIGEST_LENGTH; ++i)
        [hashStr appendFormat:@"%02x", outputData[i]];



    [innerpool drain];

    //NSLog(@"%@ hash : %@",filePath,hashStr);

    FileHash *hash = [[[FileHash alloc]init]autorelease];
    [hash setFileHash:hashStr];
    [hash setFilePath:filePath];

    return hash;    
}

我希望这会对某人有所帮助:)

谢谢您的回答。

ps:也许有人可以告诉我为什么我不能自己释放它,或者为什么保留计数为 2。

于 2009-11-18T12:39:14.917 回答