gpt4 book ai didi

javascript - 如何在 IE 中从 Javascript 访问 XHR responseBody(用于二进制数据)?

转载 作者:行者123 更新时间:2023-12-01 11:36:04 28 4
gpt4 key购买 nike

我有一个使用 XMLHttpRequest 的网页下载二进制资源。

在 Firefox 和 Gecko 中,我可以使用 responseText 来获取字节,即使字节流包含二进制零。我可能需要用 overrideMimeType() 强制 mimetype实现这一目标。但是,在 IE 中,responseText 不起作用,因为它似乎在第一个零处终止。如果您读取 100,000 个字节,并且字节 7 是二进制零,您将只能访问 7 个字节。 IE 的 XMLHttpRequest 暴露了一个 responseBody属性来访问字节。我看过一些帖子表明不可能直接从 Javascript 以任何有意义的方式访问此属性。这对我来说听起来很疯狂。
xhr.responseBody可以从 VBScript 访问,因此显而易见的解决方法是在网页中用 VBScript 定义一个方法,然后从 Javascript 调用该方法。见 jsdap举个例子。 编辑:不要使用这个 VBScript!

var IE_HACK = (/msie/i.test(navigator.userAgent) && 
!/opera/i.test(navigator.userAgent));

// no no no! Don't do this!
if (IE_HACK) document.write('<script type="text/vbscript">\n\
Function BinaryToArray(Binary)\n\
Dim i\n\
ReDim byteArray(LenB(Binary))\n\
For i = 1 To LenB(Binary)\n\
byteArray(i-1) = AscB(MidB(Binary, i, 1))\n\
Next\n\
BinaryToArray = byteArray\n\
End Function\n\
</script>');

var xml = (window.XMLHttpRequest)
? new XMLHttpRequest() // Mozilla/Safari/IE7+
: (window.ActiveXObject)
? new ActiveXObject("MSXML2.XMLHTTP") // IE6
: null; // Commodore 64?


xml.open("GET", url, true);
if (xml.overrideMimeType) {
xml.overrideMimeType('text/plain; charset=x-user-defined');
} else {
xml.setRequestHeader('Accept-Charset', 'x-user-defined');
}

xml.onreadystatechange = function() {
if (xml.readyState == 4) {
if (!binary) {
callback(xml.responseText);
} else if (IE_HACK) {
// call a VBScript method to copy every single byte
callback(BinaryToArray(xml.responseBody).toArray());
} else {
callback(getBuffer(xml.responseText));
}
}
};
xml.send('');

这是真的吗?最好的方法?复制每个字节?对于不会非常有效的大型二进制流。

还有一种使用 ADODB.Stream 的可能技术,它是 MemoryStream 的 COM 等效项。 See here例如。它不需要 VBScript,但需要一个单独的 COM 对象。
if (typeof (ActiveXObject) != "undefined" && typeof (httpRequest.responseBody) != "undefined") {
// Convert httpRequest.responseBody byte stream to shift_jis encoded string
var stream = new ActiveXObject("ADODB.Stream");
stream.Type = 1; // adTypeBinary
stream.Open ();
stream.Write (httpRequest.responseBody);
stream.Position = 0;
stream.Type = 1; // adTypeBinary;
stream.Read.... /// ???? what here
}

但这不会很好地工作 - 现在大多数机器上都禁用了 ADODB.Stream。

在 IE8 开发人员工具(相当于 Firebug 的 IE)中,我可以看到 responseBody 是一个字节数组,我什至可以看到字节本身。数据就在那里。我不明白为什么我做不到。

我可以用 responseText 阅读它吗?

提示? (除了定义一个 VBScript 方法)

最佳答案

是的,我想出的通过 IE 中的 XHR 读取二进制数据的答案是使用 VBScript 注入(inject)。一开始这让我很反感,但是,我把它看作是更多依赖于浏览器的代码。
(常规的 XHR 和 responseText 在其他浏览器中工作正常;您可能必须使用 XMLHttpRequest.overrideMimeType() 强制 mime 类型。这在 IE 上不可用)。

这就是我得到一个像 responseText 这样的东西的方式在 IE 中,即使是二进制数据。
首先,一次性注入(inject)一些 VBScript,如下所示:

if(/msie/i.test(navigator.userAgent) && !/opera/i.test(navigator.userAgent)) {
var IEBinaryToArray_ByteStr_Script =
"<!-- IEBinaryToArray_ByteStr -->\r\n"+
"<script type='text/vbscript' language='VBScript'>\r\n"+
"Function IEBinaryToArray_ByteStr(Binary)\r\n"+
" IEBinaryToArray_ByteStr = CStr(Binary)\r\n"+
"End Function\r\n"+
"Function IEBinaryToArray_ByteStr_Last(Binary)\r\n"+
" Dim lastIndex\r\n"+
" lastIndex = LenB(Binary)\r\n"+
" if lastIndex mod 2 Then\r\n"+
" IEBinaryToArray_ByteStr_Last = Chr( AscB( MidB( Binary, lastIndex, 1 ) ) )\r\n"+
" Else\r\n"+
" IEBinaryToArray_ByteStr_Last = "+'""'+"\r\n"+
" End If\r\n"+
"End Function\r\n"+
"</script>\r\n";

// inject VBScript
document.write(IEBinaryToArray_ByteStr_Script);
}

我正在使用的读取二进制文件的 JS 类公开了一个有趣的方法, readCharAt(i) ,它读取第 i 个索引处的字符(实际上是一个字节)。这就是我的设置方式:
// see doc on http://msdn.microsoft.com/en-us/library/ms535874(VS.85).aspx
function getXMLHttpRequest()
{
if (window.XMLHttpRequest) {
return new window.XMLHttpRequest;
}
else {
try {
return new ActiveXObject("MSXML2.XMLHTTP");
}
catch(ex) {
return null;
}
}
}

// this fn is invoked if IE
function IeBinFileReaderImpl(fileURL){
this.req = getXMLHttpRequest();
this.req.open("GET", fileURL, true);
this.req.setRequestHeader("Accept-Charset", "x-user-defined");
// my helper to convert from responseBody to a "responseText" like thing
var convertResponseBodyToText = function (binary) {
var byteMapping = {};
for ( var i = 0; i < 256; i++ ) {
for ( var j = 0; j < 256; j++ ) {
byteMapping[ String.fromCharCode( i + j * 256 ) ] =
String.fromCharCode(i) + String.fromCharCode(j);
}
}
// call into VBScript utility fns
var rawBytes = IEBinaryToArray_ByteStr(binary);
var lastChr = IEBinaryToArray_ByteStr_Last(binary);
return rawBytes.replace(/[\s\S]/g,
function( match ) { return byteMapping[match]; }) + lastChr;
};

this.req.onreadystatechange = function(event){
if (that.req.readyState == 4) {
that.status = "Status: " + that.req.status;
//that.httpStatus = that.req.status;
if (that.req.status == 200) {
// this doesn't work
//fileContents = that.req.responseBody.toArray();

// this doesn't work
//fileContents = new VBArray(that.req.responseBody).toArray();

// this works...
var fileContents = convertResponseBodyToText(that.req.responseBody);

fileSize = fileContents.length-1;
if(that.fileSize < 0) throwException(_exception.FileLoadFailed);
that.readByteAt = function(i){
return fileContents.charCodeAt(i) & 0xff;
};
}
if (typeof callback == "function"){ callback(that);}
}
};
this.req.send();
}

// this fn is invoked if non IE
function NormalBinFileReaderImpl(fileURL){
this.req = new XMLHttpRequest();
this.req.open('GET', fileURL, true);
this.req.onreadystatechange = function(aEvt) {
if (that.req.readyState == 4) {
if(that.req.status == 200){
var fileContents = that.req.responseText;
fileSize = fileContents.length;

that.readByteAt = function(i){
return fileContents.charCodeAt(i) & 0xff;
}
if (typeof callback == "function"){ callback(that);}
}
else
throwException(_exception.FileLoadFailed);
}
};
//XHR binary charset opt by Marcus Granado 2006 [http://mgran.blogspot.com]
this.req.overrideMimeType('text/plain; charset=x-user-defined');
this.req.send(null);
}

conversion code由 Miskun 提供。

非常快,效果很好。

我使用这种方法从 Javascript 中读取和提取 zip 文件,并在一个用 Javascript 读取和显示 EPUB 文件的类中。很合理的表现。一个 500kb 的文件大约需要半秒。

关于javascript - 如何在 IE 中从 Javascript 访问 XHR responseBody(用于二进制数据)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1919972/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com