我正在尝试计算ALAsset的CRC校验和。我的目标是保存所有CRC并在以后比较它们以查看资产是否已更改,但每次我为同一资产生成CRC时,我得到的结果都不同。
生成CRC:
#import "CRC32.h"
#import <zlib.h>
@implementation CRC32
+ (uint32_t)CRC32Value:(NSData*)data
{
uLong crc = crc32(0L, Z_NULL, 0);
crc = crc32(crc, [data bytes], [data length]);
return crc;
}
@end
如何使用:
void(^assetEnumerator)(ALAsset *, NSUInteger, BOOL *) = ^(ALAsset *result, NSUInteger index, BOOL *stop)
{
if(result == nil)
{
return;
}
CrawlAssetData *assetData = [[CrawlAssetData alloc] init];
[assetData setCrc:[CRC32 CRC32Value:[NSData dataWithBytes:&(result)
length:sizeof(result)]]];
以下是在同一资产的不同场合生成CRC时的结果:
id:17575
url:assets-library://asset/asset.JPG?id = BB282CBD-F5B1-4771-B48B-E021224C7384&amp; ext = JPG
filesize:1394332
crc:3605102491
创作时间:2456085.397025
id:17826
url:assets-library://asset/asset.JPG?id = BB282CBD-F5B1-4771-B48B-E021224C7384&amp; ext = JPG
filesize:1394332
crc:1383370697
创作时间:2456085.397025
如您所见,filesize和url是相同的,但CRC是不同的。
我计算CRC错了吗?或者我应该使用ALAsset的不同部分来生成CRC?每次检索ALAsset时,某些数据可能不同吗?
提前致谢!
答案 0 :(得分:1)
这一行:
[assetData setCrc:[CRC32 CRC32Value:[NSData dataWithBytes:&(result)
length:sizeof(result)]]];
...正在计算ALAsset实例的地址的CRC,而不是其数据。
您可以通过拆分行来验证这一点:
NSData *crcData = [NSData dataWithBytes:&(result)
length:sizeof(result)];
NSLog( @"crcData length: %d", [crcData length]);
[assetData setCrc:[CRC32 CRC32Value:crcData]];
我的猜测是,在输出中你会看到crcData length: 4
。
根据提问者的后续评论(进行更正),执行此操作的代码为:
ALAssetRepresentation *rep = [result defaultRepresentation];
uint8_t *buffer = malloc(rep.size);
NSUInteger buffered = [rep getBytes:buffer
fromOffset:0
length:rep.size
error:nil];
NSData *data = [NSData dataWithBytesNoCopy:buffer
length:buffered
freeWhenDone:YES];
uint32_t CRC32 = [CRC32 CRC32Value:data];
[assetData setCrc:CRC32];
特别要注意sizeof(NSData*)
(或NSData *a; sizeof(A)
)总是与指针的大小相同(在32位系统上为4,如iPhone,8为64位Mac OS X),而不是NSData
中存储的字节数据的长度。获取sizeof()
NSObject 指针是没有正常理由的。