如何在swift语言中使用CC_MD5方法。
在objective-c中,我们可以用下面的代码encryptionmd5的string
const char *cStr = [someString UTF8String]; unsigned char result[16]; CC_MD5( cStr, strlen(cStr), result ); md5String = [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] ];
但是现在CC_MD5不用快速工作。 如何处理这个。
这是我想出来的。 这是String的扩展。 不要忘记添加#import <CommonCrypto/CommonCrypto.h>
到Xcode创build的ObjC-Swift桥接头。
extension String { var md5: String! { let str = self.cStringUsingEncoding(NSUTF8StringEncoding) let strLen = CC_LONG(self.lengthOfBytesUsingEncoding(NSUTF8StringEncoding)) let digestLen = Int(CC_MD5_DIGEST_LENGTH) let result = UnsafeMutablePointer<CUnsignedChar>.alloc(digestLen) CC_MD5(str!, strLen, result) let hash = NSMutableString() for i in 0..<digestLen { hash.appendFormat("%02x", result[i]) } result.dealloc(digestLen) return String(format: hash as String) } }
这里是我在Swift 3.0中的版本,我相信它比其他答案更安全和更快。
带有#import <CommonCrypto/CommonCrypto.h>
桥接头是必需的。
func MD5(_ string: String) -> String? { let length = Int(CC_MD5_DIGEST_LENGTH) var digest = [UInt8](repeating: 0, count: length) if let d = string.data(using: String.Encoding.utf8) { _ = d.withUnsafeBytes { (body: UnsafePointer<UInt8>) in CC_MD5(body, CC_LONG(d.count), &digest) } } return (0..<length).reduce("") { $0 + String(format: "%02x", digest[$1]) } }
作为CryptoSwift项目的一部分,我做了纯粹的Swift的MD5实现。
我可以在这里复制代码,但是它使用了作为这个项目的一部分的扩展,所以对于复制和粘贴的使用可能是无用的。 但是,你可以看看那里并使用它。
所以这里是解决scheme,我知道这将节省您的时间100%
BridgingHeader – >用于将Objective-c代码暴露给Swift项目
CommonCrypto – >是使用md5散列所需的文件
由于Common Crypto是一个Objective-c文件,因此需要使用BridgingHeader来使用散列所需的方法
(例如)
extension String { func md5() -> String! { let str = self.cStringUsingEncoding(NSUTF8StringEncoding) let strLen = CUnsignedInt(self.lengthOfBytesUsingEncoding(NSUTF8StringEncoding)) let digestLen = Int(CC_MD5_DIGEST_LENGTH) let result = UnsafeMutablePointer<CUnsignedChar>.alloc(digestLen) CC_MD5(str!, strLen, result) var hash = NSMutableString() for i in 0..<digestLen { hash.appendFormat("%02x", result[i]) } result.destroy() return String(format: hash as String) }
}
如何将Common Crypto添加到Swift项目?
这个链接将教你如何(STEP由STEP)。
我build议使用桥接头
*************更新了Swift 3 ****************
extension String { func toMD5() -> String { if let messageData = self.data(using:String.Encoding.utf8) { var digestData = Data(count: Int(CC_MD5_DIGEST_LENGTH)) _ = digestData.withUnsafeMutableBytes {digestBytes in messageData.withUnsafeBytes {messageBytes in CC_MD5(messageBytes, CC_LONG((messageData.count)), digestBytes) } } return digestData.hexString() } return self } } extension Data { func hexString() -> String { let string = self.map{ String($0, radix:16) }.joined() return string } }
如何使用?
让stringConvertedToMD5 =“foo”.toMD5()
Xcode 6 beta 5现在使用UnsafeMutablePointer
替代UnsafePointer
。 string转换也需要format:
参数标签。
extension String { func md5() -> String! { let str = self.cStringUsingEncoding(NSUTF8StringEncoding) let strLen = CUnsignedInt(self.lengthOfBytesUsingEncoding(NSUTF8StringEncoding)) let digestLen = Int(CC_MD5_DIGEST_LENGTH) let result = UnsafeMutablePointer<CUnsignedChar>.alloc(digestLen) CC_MD5(str!, strLen, result) var hash = NSMutableString() for i in 0..<digestLen { hash.appendFormat("%02x", result[i]) } result.destroy() return String(format: hash) } }
事实上,如果你想从NSData中计算MD5,请看看这个:
func md5() -> NSData { var ctx = UnsafePointer<CC_MD5_CTX>.alloc(sizeof(CC_MD5_CTX)) CC_MD5_Init(ctx); CC_MD5_Update(ctx, self.bytes, UInt32(self.length)); let length = Int(CC_MD5_DIGEST_LENGTH) * sizeof(Byte) var output = UnsafePointer<Byte>.alloc(length) CC_MD5_Final(output, ctx); let outData = NSData(bytes: output, length: Int(CC_MD5_DIGEST_LENGTH)) output.destroy() ctx.destroy() //withUnsafePointer return outData; }
得到想法。
对于桥接头不是选项的情况(例如,在shell脚本中),可以通过NSTask
使用命令行工具/sbin/md5
:
import Foundation func md5hash(string: String) -> String { let t = NSTask() t.launchPath = "/sbin/md5" t.arguments = ["-q", "-s", string] t.standardOutput = NSPipe() t.launch() let outData = t.standardOutput.fileHandleForReading.readDataToEndOfFile() var outBytes = [UInt8](count:outData.length, repeatedValue:0) outData.getBytes(&outBytes, length: outData.length) var outString = String(bytes: outBytes, encoding: NSASCIIStringEncoding) assert(outString != nil, "failed to md5 input string") return outString!.stringByTrimmingCharactersInSet(NSCharacterSet.newlineCharacterSet()) }
用法:
let md5 = md5hash("hello world") // 5eb63bbbe01eeed093cb22bb8f5acdc3
需要导入#import <CommonCrypto/CommonCrypto.h>
到桥接头
我正在计算MD5散列,但只使用我使用的前16个字节
class func hash(data: NSData) -> String { let data2 = NSMutableData(length: Int(CC_MD5_DIGEST_LENGTH))! CC_MD5(data.bytes, CC_LONG(data.length), UnsafeMutablePointer<UInt8>(data2.mutableBytes)) let data3 = UnsafePointer<CUnsignedChar>(data2.bytes) var hash = "" for (var i = 0; i < 16; ++i) { hash += String(format: "%02X", data3[i]) } return hash }