【问题标题】:different adler32 check sum on objective-c and C#在objective-c和C#上不同的adler32校验和
【发布时间】:2013-08-29 08:32:51
【问题描述】:

我正在 Objective-c 中开发一个应用程序,它将文件和该文件的 Adler32 校验和发送到 Web 服务。

在我将文件发送到网络服务器后,回答说校验和失败。

这是我用来检查objective-c总和的代码:

//Get Asset NSData
ALAssetRepresentation *rep = [[p objectForKey:@"assetPath"] defaultRepresentation];
Byte *buffer = (Byte*)malloc(rep.size);
NSUInteger buffered = [rep getBytes:buffer fromOffset:0.0 length:rep.size error:nil];
NSData *data = [NSData dataWithBytesNoCopy:buffer length:buffered freeWhenDone:YES];

//I also need a path to the file so I have to save the NSData to a local file
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *documentsDirectory = [paths objectAtIndex:0];
NSString *localFilePath = [documentsDirectory stringByAppendingPathComponent:@"png"];
[data writeToFile:localFilePath atomically:YES];
NSString *path = localFilePath;

//Now call the Adler
NSLog(@"checksum_fast:%@", [NSString stringWithFormat:@"%u", adler32_fast([data bytes], [data length])]);
NSLog(@"checksum:%@", [NSString stringWithFormat:@"%u", adler32([data bytes], [data length])]);

//Adler Lib
#define MOD_ADLER 65521
#define BUFFER_SIZE 1000

uint32_t adler32_update(uint32_t ck, const unsigned char data) {
  uint32_t a = ck & 0x0ffff, b = (ck >> 16) & 0x0ffff;
  a = (a + data) % MOD_ADLER;
  b = (b + a) % MOD_ADLER;
  return (b << 16) | a;
}

uint32_t adler32(const unsigned char *data, int data_len) {
  uint64_t ck = 1;
  for(int i=0; i<data_len; i++) {
    ck = adler32_update(ck,data[i]);
  }
  return ck;
}

uint32_t adler32_fast(const unsigned char *data, int data_len) {
  uint32_t a = 1, b = 0;
  for(int i=0; i<data_len; i++) {
    a = (a + data[i]) % MOD_ADLER;
    b = (b + a) % MOD_ADLER;
  }
  return (b << 16) | a;
}

而这一段是C# web服务的代码:

//It use ICSharpCode.SharpZipLib.dll    
byte[] buff = null;
FileStream fs = new FileStream(path, FileMode.Open, FileAccess.Read);
BinaryReader br = new BinaryReader(fs);
long numBytes = new FileInfo(path).Length;
buff = br.ReadBytes((int)numBytes);

Adler32 adler = new Adler32();
adler.Reset();
adler.Update(buff, 0, buff.Length);
fs.Close();
fs.Dispose();
br.Close();
buff = null;
return Convert.ToString(adler.Value, 16);

有人知道为什么我会得到不同的校验和结果吗?

测试文件:http://we.tl/gjqSj43238

Adler32 C#: 701ea682 (1881056898)

Adler32 Obj-C: 70100C51 (1880099921)

谢谢!

【问题讨论】:

  • 701ea682 是正确的校验值。
  • 当我直接在数据上使用你的adler32_fast() 时,它给出了正确的答案。您一定是错误地加载了data

标签: c# objective-c adler32


【解决方案1】:

最重要的是,您拼错了 Adler。你的常量应该是MOD_ADLER,而不是MOD_ALDER

您还没有提供所有代码,所以我不知道您将该常量设置为什么。

您的代码并不像名称所声称的那样“快速”,因为您每次都进行模运算。如果您不打算推迟模运算,那么在那里有两个嵌套循环是没有意义的。

1 &amp; 0xFFFF 看起来很傻,因为那总是1

您的错误是您将b 初始化为1。它应该被初始化为0

啊,我看到另一个错误:您的数据数组需要使用unsigned char,而不是char

我建议你复制adler32() implementation in zlib

【讨论】:

  • 谢谢 Mark,MOD_ADLER 的名字是固定的,值是 65521。另外我贴了更多的代码,就这些了。
  • 现在你的代码会给出正确的检查值,虽然会很慢。
  • 每个代码的结果都不同,我尝试了一个计算 Adler32 的唯一工具,C# 代码具有相同的结果,但在 obj-c 中我得到了其他结果。 C#: 1881056898 Obj-c: 1880099921 在线工具: 1881056898 可能问题是当我将 [data bytes] 传递给函数时,它需要 const char *data。
  • 嗨,马克,它不起作用,现在我进行了更改,我得到了 3243175871。谢谢。
  • 标记问题已更新为我正在使用的文件以及 NSData 的来源。
猜你喜欢
  • 2014-03-02
  • 1970-01-01
  • 2020-07-02
  • 2017-03-09
  • 2011-05-06
  • 2010-11-21
  • 1970-01-01
  • 1970-01-01
  • 2012-03-30
相关资源
最近更新 更多