Алгоритм MD5 в Objective-C

Ответы:

219

md5 доступен на iPhone и может быть добавлен как дополнение для ie NSStringи NSDataкак ниже.

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/

epatel
источник
Это вытягивает весь файл в память?
openfrog
Это не о файлах. Если вы хотите создать MD5 из файла с помощью этих методов, то вы можете выполнить NSData * fileContents = [NSData dataWithContentsOfFile: @ "<yourPath>"]; NSString * myHash = [fileContents md5]; И да, это вытянет весь файл в память. Если вы найдете решение, которое работает с файловыми потоками, опубликуйте его как ответ.
Клаас
1
Если вам нужен хеш-файл, вы должны использовать CC_MD5_Init, затем CC_MD5_Update для всех данных файла, а после этого - CC_MD5_Finish.
Николай Ольшевский
7
Компиляция для 64-битной архитектуры, вызов к strlenвыводит предупреждение: «Неявное преобразование теряет целочисленную точность:« unsigned long »в« CC_LONG »(он же« 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)
Нилеш Кумар
проверка @wimcNilesh selfперед выполнением; если «я» равно нулю, оно рухнет.
brandonscript
4
Этот ответ намного чище, чем остальные; одна вещь, в которой он нуждается, - это приведение, например, (int)до ...strlen(int)strlen
brandonscript
Hay Это хороший +1 upvote, и не могли бы вы также предоставить метод расшифровки md5 такой же, как ваш шифрования.
Аяз
@ Ayaz MD5 не может быть расшифрован (по крайней мере, просто с помощью метода).
albanx
9

Если производительность важна, вы можете использовать эту оптимизированную версию. Это примерно в 5 раз быстрее, чем с stringWithFormatили NSMutableString.

Это категория 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

Хорошо, так как люди просили версию потока файла. Я изменил симпатичный небольшой фрагмент, созданный Джоэлем Лопесом Да Силвой, который работает с MD5, SHA1 и SHA512, и использует потоковые потоки. Он сделан для iOS, но работает с минимальными изменениями в OSX, а также (удалите метод ALAssetRepresentation). Он может создавать контрольные суммы для файлов с заданным путем к файлу или ALAssets (используя ALAssetRepresentation). Он разбивает данные на небольшие пакеты, делая воздействие на память минимальным независимо от размера файла / размера ресурса.

В настоящее время он находится на github здесь: https://github.com/leetal/FileHash

Александр W
источник
Код, опубликованный Джоэлом, имеет состояние гонки, и, похоже, ваш может его унаследовать. Смотрите комментарий, который я опубликовал на пост Джоэла. joel.lopes-da-silva.com/2010/09/07/…
xyzzycoder
Спасибо! Исправил это сейчас. Это никогда не было проблемой для меня, так как в оригинальной реализации я всегда запускал его в отдельной ветке;)
Александр W
0

Любая причина не использовать реализацию Apple: https://developer.apple.com/library/mac/documentation/Security/Conceptual/cryptoservices/GeneralPurposeCrypto/GeneralPurposeCrypto.html#//apple_ref/doc/uid/TP40011172-CH9-S1

Найдите руководство по службам криптографии на сайте разработчиков Apple.

vpathak
источник
Ссылка охватывает Common Crypto, который использует большинство ответов здесь.
zaph
1
Конечно, алгоритм такой же. Но обратите внимание, что реализация собственного криптоалгоритма может привести к недостаткам. Требуется много усилий, чтобы сделать это правильно во всех сценариях. Поэтому в общем случае предпочтительнее использовать версию библиотеки.
впатхак