Python Swift中的十六进制/二进制字符串转换

Python Swift中的十六进制/二进制字符串转换,python,swift,hex,bin,Python,Swift,Hex,Bin,Python有两个非常有用的库方法binascii.a2b_hexkeyStr和binascii.hexlifykeyBytes,我一直在Swift中使用它们。斯威夫特有什么现成的吗。如果没有,将如何实施?假设所有边界和其他检查(如偶数长度键)都已完成。不太熟悉Python及其在转换数字时执行的检查,但您可以扩展以下功能: func convert(_ str: String, fromRadix r1: Int, toRadix r2: Int) -> String? { if

Python有两个非常有用的库方法binascii.a2b_hexkeyStr和binascii.hexlifykeyBytes,我一直在Swift中使用它们。斯威夫特有什么现成的吗。如果没有,将如何实施?假设所有边界和其他检查(如偶数长度键)都已完成。

不太熟悉Python及其在转换数字时执行的检查,但您可以扩展以下功能:

func convert(_ str: String, fromRadix r1: Int, toRadix r2: Int) -> String? {
    if let num = Int(str, radix: r1) {
        return String(num, radix: r2)
    } else {
        return nil
    }
}

convert("11111111", fromRadix: 2, toRadix: 16)
convert("ff", fromRadix: 16, toRadix: 2)
斯威夫特2

extension NSData {
    class func dataFromHexString(hex: String) -> NSData? {
        let regex = try! NSRegularExpression(pattern: "^[0-9a-zA-Z]*$", options: .CaseInsensitive)
        let validate = regex.firstMatchInString(hex, options: NSMatchingOptions.init(rawValue: 0), range: NSRange(location: 0, length: hex.characters.count))
        if validate == nil || hex.characters.count % 2 != 0 {
            return nil
        }
        let data = NSMutableData()
        for i in 0..<hex.characters.count/2 {
            let hexStr = hex.substring(i * 2, length: 2)
            var ch: UInt32 = 0
            NSScanner(string: hexStr).scanHexInt(&ch)
            data.appendBytes(&ch, length: 1)
        }
        return data
    }
}

let a = 0xabcd1234
print(String(format: "%x", a)) // Hex to String
NSData.dataFromHexString("abcd1234") // String to hex
来自Swift 3的数据没有内置方法将其内容打印为 十六进制字符串,或从十六进制字符串创建数据值

十六进制字符串方法的数据可在或处找到。以下是第一个链接的实现:

extension Data {
    func hexEncodedString() -> String {
        return map { String(format: "%02hhx", $0) }.joined()
    }
}
下面是将十六进制字符串反向转换为数据的可能实现 代码评审时进行的转换,翻译成Swift 3并改进 作为一个可失败的初始化器:

extension Data {

    init?(fromHexEncodedString string: String) {

        // Convert 0 ... 9, a ... f, A ...F to their decimal value,
        // return nil for all other input characters
        func decodeNibble(u: UInt16) -> UInt8? {
            switch(u) {
            case 0x30 ... 0x39:
                return UInt8(u - 0x30)
            case 0x41 ... 0x46:
                return UInt8(u - 0x41 + 10)
            case 0x61 ... 0x66:
                return UInt8(u - 0x61 + 10)
            default:
                return nil
            }
        }

        self.init(capacity: string.utf16.count/2)
        var even = true
        var byte: UInt8 = 0
        for c in string.utf16 {
            guard let val = decodeNibble(u: c) else { return nil }
            if even {
                byte = val << 4
            } else {
                byte += val
                self.append(byte)
            }
            even = !even
        }
        guard even else { return nil }
    }
}

@这些答案不适用于大数字。我说的是32位十六进制数字。有效的方法是使用如此大的字符串并将输入转换为字节数组。Swift stdlib目前还没有BigInteger和类似的转换。这应该得到1000000分。超级干净和有用。谢谢分享!
// Hex string to Data:
if let data = Data(fromHexEncodedString: "0002468A13579BFF") {
    let idata = Data(data.map { 255 - $0 })

    // Data to hex string:
    print(idata.hexEncodedString()) // fffdb975eca86400
} else {
    print("invalid hex string")
}