11

我正在尝试在 IE 上做跨域。

我使用 XDomainRequest,并为所有事件(onerror、onload、onprogress 和 ontimeout)植入日志以监控进度。

它有时会起作用,但并非总是如此(一台计算机,IE9,相同的站点,相同的请求,3 或 4 个中的 1 个有效;另一台计算机,IE8,可能 2 个中的 1 个有效)。我没有从日志中得到任何有用的信息,因为没有触发任何东西。

我很困扰。IE有什么调试工具吗?为什么有时 XDomainRequest 不起作用?

非常感谢科罗宁

4

2 回答 2

19

XDomainRequest 对象中至少有两个重大错误,一个影响 IE8,另一个影响 IE9。

问题 1 - 垃圾收集

在 Internet Explorer 8 中,在调用 send() 但尚未完成后,XDomainRequest 对象错误地受到垃圾回收的影响。此错误的症状是开发人员工具的网络跟踪显示请求“已中止”,并且没有调用任何错误、超时或成功事件处理程序。

典型的 AJAX 代码看起来有点像这样:

function sendCrossDomainAjax(url, successCallback, errorCallback) {
  var xdr = new XDomainRequest();
  xdr.open("get", url);
  xdr.onload = function() { successCallback(); }
  xdr.onerror = function() { errorCallback(); }
  xdr.send();
}

在此示例中,包含 XDomainRequest 的变量超出范围。如果用户运气不好,IE 的 Javascript 垃圾收集器将在 send() 异步完成之前运行,请求将被中止。即使 XDomainRequest 对象可以被捕获到 OnLoad 和 OnError 事件处理程序中,IE 也会看到整个对象图没有对它的引用,并将垃圾收集它。IE 应该“固定”对象直到完成。

您会注意到互联网上的许多其他讨论都提到在 xdr.send(); 周围放置一个 setTimeout;call 将以某种方式“解决”神秘的 XDomainRequest 失败。这是一个kludge,而且完全不正确。正在发生的所有事情是 XDomainRequest 对象被“固定”到 setTimeout 闭包中,并且不会很快受到垃圾回收的影响。它不能解决问题。

要正确解决此问题,请确保 XDomainRequest 存储在全局变量中,直到请求完成。例如:

var pendingXDR = [];

function removeXDR(xdr) {
  // indexOf isn't always supported, you can also use jQuery.inArray()
  var index = pendingXDR.indexOf(xdr);
  if (index >= 0) {
    pendingXDR.splice(index, 1);
  }
}

function sendCrossDomainAjax(url, successCallback, errorCallback) {
  var xdr = new XDomainRequest();
  xdr.open("get", url);

  xdr.onload = function() {
    removeXDR(xdr);
    successCallback();
  }

  xdr.onerror = function() {
    removeXDR(xdr);
    errorCallback();
  }

  xdr.send();
  pendingXDR.push(xdr);
}

问题 2 - 缺少 OnProgress 事件处理程序

第二个问题是已知的。Internet Explorer 9 在 XDomainRequest 对象中引入了回归,其中缺少(空)OnProgress 事件处理程序将导致请求在尝试报告进度信息时中止。

对于快速请求,IE9 从不尝试调用 OnProgress 事件处理程序并且请求成功。某些情况,例如当 IE 由于打开的连接过多、网络延迟、服务器响应缓慢或请求或响应负载过大而延迟请求时,会导致 IE9 开始报告进度信息。

IE9 尝试调用事件处理程序而不首先检查它是否存在,并且 XDomainRequest 对象在内部崩溃并自行销毁。

要解决此问题,请始终确保将事件处理程序附加到 OnProgress。鉴于该错误,将事件处理程序防御性地添加到所有对象的事件中并不是一个坏主意。

var xdr = new XDomainRequest();
xdr.open("get", url);
xdr.onprogress = function() { };
// regsister other event handlers

其他问题

我似乎报告说,如果在调用 .open() 之前注册事件处理程序,XDomainRequest 可能会失败。同样,在防守端,在 .open() 和 .send() 调用之间注册它们并不是一个坏主意。我还没有亲自验证这是否是一个实际的错误。

如果您遇到“拒绝访问”错误,那是因为 XDomainRequest 不允许目标页面和主机页面之间的 URI 方案不匹配。换句话说,尽量不要从 HTTPS 页面调用 HTTP 资源。

请注意 Internet 上的大多数 XDomainRequest 库。我查看了大多数流行的插件,例如各种 jQuery AJAX 传输插件(包括此处另一个答案中链接的插件)。

当然,XDomainRequest 受制于它的所有正常限制和约束。这些本身并不是错误,与替代方案(iframe kludges、Flash crossdomain.xml 传输)相比,它们并没有那么糟糕。

我在这里发布了一个公共域许可下的新 jQuery AJAX XDomainRequest 传输:https ://github.com/ebickle/snippets/tree/master/javascript/xdomainrequest

于 2014-12-20T18:42:06.577 回答
11

有完全相同的问题。简短的解决方案:

  1. 使用此代码:https ://github.com/jaubourg/ajaxHooks/blob/master/src/ajax/xdr.js

更新:链接已损坏,请在此处找到 jaubourgs 修复:https ://github.com/jaubourg/ajaxHooks/blob/master/src/xdr.js

  1. 添加xdr.onprogress = function() {};该 xdr.js 文件

可以在此处的 jQuery 主题讨论中找到详细信息

http://bugs.jquery.com/ticket/8283

其中最后一个回复包括源自此错误讨论的 xdr.onprogress 修复,该错误讨论的标题恰如其分

“如果未指定所有事件处理程序,IE9 RTM - XDomainRequest 发出的请求可能会中止” http://social.msdn.microsoft.com/Forums/en-US/iewebdevelopment/thread/30ef3add-767c-4436-b8a9-f1ca19b4812e

于 2012-03-28T09:57:49.787 回答