5

我正在尝试散列一个字符串。但是 IE11 和 Safari 中的 TextEncoder 函数的替代方法是什么?

var string = "foobar";

window.crypto.subtle.digest(

    { "name": "SHA-256" },

    new TextEncoder("utf-8").encode(string)).then(function (hash)
    {
        console.log(hex(hash)); // 'c3ab8ff13720e8ad9047dd39466b3c8974e592c2fa383d4a3960714caef0c4f2'
    }
);

var string = "foobar";

var buffer = new TextEncoder("utf-8").encode(string); // Uint8Array (ArrayBuffer)
var string = new TextDecoder("utf-8").decode(buffer); // string

console.log("buffer", buffer);
console.log("string '" + string + "'");

hex 是我从 Mozilla 得到的一个函数

https://developer.mozilla.org/en-US/docs/Web/API/SubtleCrypto/digest

4

1 回答 1

8

编辑:我刚刚使用了http://www.movable-type.co.uk/scripts/sha256.html中的 SHA256 例程,因为我不需要高性能,而且代码又短又甜,而且代码相同到处都在使用路径。事实证明,crypto.subtle API 有太多对我来说失败的极端情况(我在一些移动设备上遇到了问题,除了下面链接的 Chrome 问题让我很伤心,而且我也需要支持旧版浏览器)。

我为 IE11 编写了这个对我有用的函数(虽然几乎没有经过测试):

    function textEncode(str) {
        if (window.TextEncoder) {
            return new TextEncoder('utf-8').encode(str);
        }
        var utf8 = unescape(encodeURIComponent(str));
        var result = new Uint8Array(utf8.length);
        for (var i = 0; i < utf8.length; i++) {
            result[i] = utf8.charCodeAt(i);
        }
        return result;
    }

请注意还有其他问题,因为 IE11 没有承诺,并且msCrypto.subtle.digest()是同步的。以下内容可能对您有用,尽管您应该修复这些 hack(它需要工作以使其健壮并使用 Promise polyfill):

function sha256(str) {
    function hex(buffer) {
        var hexCodes = [];
        var view = new DataView(buffer);
        for (var i = 0; i < view.byteLength; i += 4) {
            var value = view.getUint32(i);
            var stringValue = value.toString(16);
            var padding = '00000000';
            var paddedValue = (padding + stringValue).slice(-padding.length);
            hexCodes.push(paddedValue);
        }
        return hexCodes.join('');
    }
    var buffer = textEncode(str);
    var res = crypto.subtle.digest('SHA-256', buffer);
    if (res.then) {
        return res.then(function (hash) {
            return hex(hash);
        });
    } else if (res.result) {    // IE11
        return {
            then: function(resolver) {
                resolver(hex(res.result));
            }
        }
    }
}

还要注意这个 Chrome 问题,您可以使用正则表达式/^https:|^file:|^http:\/\/localhost|^http:\/\/127.0.0.1/.test(location.protocol)进行测试。

于 2016-04-01T03:01:06.203 回答