我正在研究Objective C Cocoa应用程序。我测试CC_MD5 in CommonCrypto,它工作得很好;然而,当我给了它5G的文件时,我的整个计算机冻结并坠毁。 MD5算法将输入处理为512字节的块,并不一次要求全部输入。目标C或C中是否有一个库要求下一个512字节块,而不是一次采用所有输入?是否有一个MD5库不需要同时输入整个输入?
9
A
回答
11
还有就是在OBJ-C这里计算的大型文件MD5一个伟大的线程: http://www.iphonedevsdk.com/forum/iphone-sdk-development/17659-calculating-md5-hash-large-file.html
下面是解有人想出有:
+(NSString*)fileMD5:(NSString*)path
{
NSFileHandle *handle = [NSFileHandle fileHandleForReadingAtPath:path];
if(handle== nil) return @"ERROR GETTING FILE MD5"; // file didnt exist
CC_MD5_CTX md5;
CC_MD5_Init(&md5);
BOOL done = NO;
while(!done)
{
NSAutoreleasePool * pool = [NSAutoreleasePool new];
NSData* fileData = [handle readDataOfLength: CHUNK_SIZE ];
CC_MD5_Update(&md5, [fileData bytes], [fileData length]);
if([fileData length] == 0) done = YES;
[pool drain];
}
unsigned char digest[CC_MD5_DIGEST_LENGTH];
CC_MD5_Final(digest, &md5);
NSString* s = [NSString stringWithFormat: @"%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x",
digest[0], digest[1],
digest[2], digest[3],
digest[4], digest[5],
digest[6], digest[7],
digest[8], digest[9],
digest[10], digest[11],
digest[12], digest[13],
digest[14], digest[15]];
return s;
}
2
CC_MD5()
被设计用来处理所有它的输入一次。 5GB可能比实际存储在任何地方更多。对于更大的数据,如果您使用CC_MD5_CTX
,CC_MD5_Init()
,CC_MD5_Update()
和CC_MD5_Final()
,CommonCrypto可以一次对它进行大块操作。查看CommonCrypto文档或Google获取更多信息和示例代码。
2
这是一个更好的方法来使用dispatch apis来做到这一点,以获得更高的效率。我在生产中使用它,它工作正常!
#import "CalculateMD5.h"
// Cryptography
#include <CommonCrypto/CommonDigest.h>
@implementation CalculateMD5
- (id)init
{
self = [super init];
if (self)
{
MD5ChecksumOperationQueue = dispatch_queue_create("com.test.calculateMD5Checksum", DISPATCH_QUEUE_SERIAL);
}
return self;
}
- (void)closeReadChannel
{
dispatch_async(MD5ChecksumOperationQueue, ^{
dispatch_io_close(readChannel, DISPATCH_IO_STOP);
});
}
- (void)MD5Checksum:(NSString *)pathToFile TCB:(void(^)(NSString *md5, NSError *error))tcb
{
// Initialize the hash object
__block CC_MD5_CTX hashObject;
CC_MD5_Init(&hashObject);
readChannel = dispatch_io_create_with_path(DISPATCH_IO_STREAM,
pathToFile.UTF8String,
O_RDONLY, 0,
MD5ChecksumOperationQueue,
^(int error) {
[self closeReadChannel];
});
if (readChannel == nil)
{
NSError* e = [NSError errorWithDomain:@"MD5Error"
code:-999 userInfo:@{
NSLocalizedDescriptionKey : @"failed to open file for calculating MD5."
}];
tcb(nil, e);
return;
}
dispatch_io_set_high_water(readChannel, 512*1024);
dispatch_io_read(readChannel, 0, SIZE_MAX, MD5ChecksumOperationQueue, ^(bool done, dispatch_data_t data, int error) {
if (error != 0)
{
NSError* e = [NSError errorWithDomain:@"ExamSoftMD5"
code:error userInfo:@{
NSLocalizedDescriptionKey : @"failed to read from file for calculating MD5."
}];
tcb(nil, e);
[self closeReadChannel];
return;
}
if (dispatch_data_get_size(data) > 0)
{
const void *buffer = NULL;
size_t size = 0;
data = dispatch_data_create_map(data, &buffer, &size);
CC_MD5_Update(&hashObject, (const void *)buffer, (CC_LONG)size);
}
if (done == YES)
{
// Compute the hash digest
unsigned char digest[CC_MD5_DIGEST_LENGTH];
CC_MD5_Final(digest, &hashObject);
// Compute the string result
char *hash = calloc((2 * sizeof(digest) + 1), sizeof(char));
for (size_t i = 0; i < sizeof(digest); ++i)
{
snprintf(hash + (2 * i), 3, "%02x", (int)(digest[i]));
}
tcb(@(hash), nil);
[self closeReadChannel];
}
});
}
@end
相关问题
- 1. 询问用户是否需要输入多个输入
- 2. 是否有一个用于实时视频输入的Python库?
- 3. Cin需要2个输入
- 4. 将两个不同的输出输入到一个需要两个输入的命令中
- 5. 我需要输入一个数字。如果不是,显示请输入数字,并再次要求输入
- 6. 如何在一个输入有两个输出时将一个输入限制为一个输出?数据库
- 7. 获取一个unicode输入,需要它是一个字符串
- 8. 是否需要输入验证?
- 9. JSON输入过滤是否需要
- 10. 是否需要密码输入消毒?
- 11. Ç程序 - 需要,同时用户输入的输入
- 12. 需要输入
- 13. 在一个输入表格中输入两个不同的值
- 14. 插入Cassandra时是否需要转义数据输入?
- 15. 确定一个pty是否有编程要求输入
- 16. 我需要一个功能,我可以禁止字符输入,我只需要用户的整数输入
- 17. jQuery来限制我有一个文本框,输入需要输入
- 18. 编写一个输入验证循环,要求用户输入“是”或“否”
- 19. 同时输入两个uitextfield
- 20. 检查输入是否已经输入到一个数组中
- 21. 如何检查是否输入了一个值使用输入
- 22. 检查用户输入是否是一个整数c#
- 23. C++函数检查输入是否是一个整数?
- 24. 检查输入是否是一个整数C
- 25. Angular 2 Custom Validator:检查输入值是否是一个整数?
- 26. 校验和同时输入一个输入
- 27. RoR-检查输入是否是MD5散列或不是?
- 28. 为什么有一个文本输入的FORM输入,而有两个文本INPUT的输入不能输入?
- 29. 需要检查输入是否是小数或不是
- 30. Java获取整数输入法需要输入两次
感谢您的链接,现在阅读它。 –
这个答案会从ARC的更新中受益。循环中的代码需要放在'@autoreleasepool {}'块中。 –