อัลกอริทึม MD5 ใน Objective-C


คำตอบ:


219

md5 มีอยู่ใน iPhone และสามารถเพิ่มเป็นส่วนเสริมสำหรับ ie NSStringและNSDatalike ด้านล่าง

MyAdditions.h

@interface NSString (MyAdditions)
- (NSString *)md5;
@end

@interface NSData (MyAdditions)
- (NSString*)md5;
@end

MyAdditions.m

#import "MyAdditions.h"
#import <CommonCrypto/CommonDigest.h> // Need to import for CC_MD5 access

@implementation NSString (MyAdditions)
- (NSString *)md5
{
    const char *cStr = [self UTF8String];
    unsigned char result[CC_MD5_DIGEST_LENGTH];
    CC_MD5( cStr, (int)strlen(cStr), result ); // This is the md5 call
    return [NSString stringWithFormat:
        @"%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x",
        result[0], result[1], result[2], result[3], 
        result[4], result[5], result[6], result[7],
        result[8], result[9], result[10], result[11],
        result[12], result[13], result[14], result[15]
        ];  
}
@end

@implementation NSData (MyAdditions)
- (NSString*)md5
{
    unsigned char result[CC_MD5_DIGEST_LENGTH];
    CC_MD5( self.bytes, (int)self.length, result ); // This is the md5 call
    return [NSString stringWithFormat:
        @"%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x",
        result[0], result[1], result[2], result[3], 
        result[4], result[5], result[6], result[7],
        result[8], result[9], result[10], result[11],
        result[12], result[13], result[14], result[15]
        ];  
}
@end

แก้ไข

เพิ่ม NSData md5 เพราะฉันต้องการมันเองและคิดว่านี่เป็นสถานที่ที่ดีในการบันทึกตัวอย่างเล็ก ๆ น้อย ๆ นี้ ...

วิธีการเหล่านี้ได้รับการตรวจสอบโดยใช้เวกเตอร์ทดสอบ NIST MD5 ใน http://www.nsrl.nist.gov/testdata/


สิ่งนี้ดึงไฟล์ทั้งหมดไปไว้ในหน่วยความจำหรือไม่?
openfrog

นี่ไม่เกี่ยวกับไฟล์ หากคุณต้องการสร้าง MD5 จากไฟล์ด้วยวิธีการเหล่านี้คุณสามารถทำ NSData * fileContents = [NSData dataWithContentsOfFile: @ "<yourPath>"]; NSString * myHash = [fileContents md5]; และใช่สิ่งนี้จะดึงไฟล์ทั้งหมดไปไว้ในหน่วยความจำ หากคุณพบโซลูชันที่ใช้ได้กับสตรีมไฟล์โปรดโพสต์เป็นคำตอบ
Klaas

1
หากคุณต้องการแฮชไฟล์คุณควรใช้ CC_MD5_Init จากนั้น CC_MD5_Update สำหรับข้อมูลไฟล์ทั้งหมดและหลังจากนั้น - CC_MD5_Finish
Nickolay Olshevsky

7
การคอมไพล์สำหรับสถาปัตยกรรม 64 บิตการเรียกเพื่อstrlenให้คำเตือน: "การแปลงโดยนัยสูญเสียความแม่นยำจำนวนเต็ม: 'unsigned long' ถึง 'CC_LONG' (aka 'unsigned int')"
MaxGabriel

55

คุณสามารถใช้ไลบรารี Common Crypto ในตัวเพื่อทำเช่นนั้น อย่าลืมนำเข้า:

#import <CommonCrypto/CommonDigest.h>

แล้ว:

- (NSString *) md5:(NSString *) input
{
    const char *cStr = [input UTF8String];
    unsigned char digest[CC_MD5_DIGEST_LENGTH];
    CC_MD5( cStr, strlen(cStr), digest ); // This is the md5 call

    NSMutableString *output = [NSMutableString stringWithCapacity:CC_MD5_DIGEST_LENGTH * 2];

    for(int i = 0; i < CC_MD5_DIGEST_LENGTH; i++)
    [output appendFormat:@"%02x", digest[i]];

    return  output;
}

ฉันใช้รหัสข้างต้น แต่ในขณะที่เรียกใช้แอปพลิเคชันมันขัดข้อง (CC_MD5 (cStr, strlen (cStr), ไดเจสต์) ----> บรรทัดนี้มีข้อยกเว้นในการกล่าวว่า EXC_BAD_ACCESS)
Nilesh Kumar

@wimcNilesh ตรวจสอบselfก่อนดำเนินการ; ถ้าตัวเองเป็นศูนย์มันจะพัง
brandonscript

4
คำตอบนี้อ่านสะอาดกว่าข้ออื่น ๆ มาก สิ่งหนึ่งที่ต้องการคือการโยนไป(int)ก่อนstrlenเช่น(int)strlen...
brandonscript

Hay นี่เป็นการโหวต +1 ที่ดีและคุณช่วยระบุวิธีการถอดรหัส md5 แบบเดียวกับการเข้ารหัสของคุณได้ไหม
Ayaz

@Ayaz MD5 ไม่สามารถถอดรหัสได้ (อย่างน้อยก็ใช้วิธีการ)
albanx

9

หากประสิทธิภาพเป็นสิ่งสำคัญคุณสามารถใช้เวอร์ชันที่ปรับให้เหมาะสมนี้ได้ มันเป็นเรื่องเกี่ยวกับ 5 ครั้งเร็วกว่าคนที่มีหรือstringWithFormatNSMutableString

นี่คือหมวดหมู่ของ NSString

- (NSString *)md5
{
    const char* cStr = [self UTF8String];
    unsigned char result[CC_MD5_DIGEST_LENGTH];
    CC_MD5(cStr, strlen(cStr), result);

    static const char HexEncodeChars[] = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' };
    char *resultData = malloc(CC_MD5_DIGEST_LENGTH * 2 + 1);

    for (uint index = 0; index < CC_MD5_DIGEST_LENGTH; index++) {
        resultData[index * 2] = HexEncodeChars[(result[index] >> 4)];
        resultData[index * 2 + 1] = HexEncodeChars[(result[index] % 0x10)];
    }
    resultData[CC_MD5_DIGEST_LENGTH * 2] = 0;

    NSString *resultString = [NSString stringWithCString:resultData encoding:NSASCIIStringEncoding];
    free(resultData);

    return resultString;
}

0

เนื่องจากมีคนถามหาเวอร์ชันสตรีมไฟล์ ฉันได้แก้ไขตัวอย่างเล็ก ๆ น้อย ๆ ที่สวยงามที่สร้างโดย Joel Lopes Da Silva ที่ทำงานร่วมกับ MD5, SHA1 และ SHA512 และใช้สตรีม สร้างขึ้นสำหรับ iOS แต่ใช้งานได้กับการเปลี่ยนแปลงเพียงเล็กน้อยบน OSX เช่นกัน (ลบวิธี ALAssetRepresentation) สามารถทำการตรวจสอบไฟล์ที่กำหนด filepath หรือ ALAssets (โดยใช้ ALAssetRepresentation) เป็นการรวมข้อมูลเป็นแพ็กเกจขนาดเล็กทำให้หน่วยความจำมีผลกระทบน้อยที่สุดโดยไม่คำนึงถึงขนาดไฟล์ / ขนาดเนื้อหา

ปัจจุบันอยู่ใน github ที่นี่: https://github.com/leetal/FileHash


รหัสที่ Joel เผยแพร่มีสภาพการแข่งขันและดูเหมือนว่ารหัสของคุณอาจได้รับมรดก ดูความคิดเห็นที่ฉันเผยแพร่ในโพสต์ของ Joel joel.lopes-da-silva.com/2010/09/07/…
xyzzycoder

ขอบคุณ! แก้ไขแล้ว นี่ไม่เคยเป็นปัญหาสำหรับฉันเลยตั้งแต่ในการใช้งานดั้งเดิมฉันมักจะเรียกใช้ในเธรดเฉพาะ;)
Alexander W

0

เหตุผลใดก็ตามที่ไม่ใช้การใช้งาน Apple: https://developer.apple.com/library/mac/documentation/Security/Conceptual/cryptoservices/GeneralPurposeCrypto/GeneralPurposeCrypto.html#//apple_ref/doc/uid/TP40011172-CH9-SW1

ค้นหา Cryptographic Services Guide บนไซต์นักพัฒนา Apple


ลิงก์นี้ครอบคลุมถึง Common Crypto ซึ่งคำตอบส่วนใหญ่ใช้
zaph

1
แน่นอนว่าอัลโกเหมือนกัน แต่โปรดทราบว่าการใช้อัลกอริทึมการเข้ารหัสลับของคุณเองอาจทำให้เกิดข้อบกพร่องได้ ต้องใช้เวลาในการชุบแข็งอย่างมากเพื่อให้ถูกต้องในทุกสถานการณ์ ดังนั้นการใช้ไลบรารีเวอร์ชันจึงเป็นที่ต้องการในกรณีทั่วไป
vpathak
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.