【问题标题】:What happens when no response is received for a request? I'm seeing retries当请求没有收到响应时会发生什么?我看到重试
【发布时间】:2012-12-27 11:20:19
【问题描述】:

我认为我的问题可能更多是与浏览器相关的问题,但这是一个非常基本的问题,我也想在我冒险构建 Web 应用程序时找到答案。

在我的客户端代码中,我正在调用$.ajax。此帖子可能需要一段时间才能回复。我看到的是在一段时间后再次发送请求。

我以为是我的$.ajax调用又发送了一次,但是无论我在服务器上看到多少次POST请求,我都只看到beforeSend回调调用了一次。我很确定我的代码不会多次发送,所以我认为是浏览器重试?

我知道我的服务器收到的请求不止一次,因为我运行 Wireshark 并且可以多次看到发布请求。所以我的假设是这与 HTTP 有关吗?即,如果在一定时间内未收到响应,则重新发送请求?

下面是我的通话示例。

$.ajax({
                    async: false,
                    type: 'POST',
                    url: '<%= url_for('importDevice') %>',
                    data: { device: val },
                    retryLimit: 0,
                    //callback
                    success: function(data) {
                    alert('calling import');
                    if ( data == 'nomaster')
                    {
                            // Display a warning  toast, with a title
                            toastr.warning('You must set the Master Key first!', 'Warning');
                            $.ismasterset = false;
                            //reset the form contents after added
                    } else
                    {
                            $("div#content").html(data);
                    }
                    },
                     beforeSend: function(){
                    alert('in before send');
                    }
            });

这是所有相关代码,“retryLimit”没有被使用,我只是没有从我的代码中删除它,是的,在我放入之前问题就存在了。

使用来自客户端和服务器的输出进行编辑。

好的,我安装了“用于 Firefox 的 Live Http 标头”。

在“生成器”选项卡中,我看到了一个调用

'#request# POST http://testhost/importdevice' 

我在“标题”部分没有看到POST,可能是因为没有响应?

在我的网络服务器中,虽然我看到 2 个调用相隔大约 22 秒。

[Sun Jan 13 03:08:45 2013] [debug] POST /importdevice (Mozilla/5.0 (Macintosh; Intel Mac OS X 10.8; rv:17.0) Gecko/20100101 Firefox/17.0).

[Sun Jan 13 03:09:07 2013] [debug] POST /importdevice (Mozilla/5.0 (Macintosh; Intel Mac OS X 10.8; rv:17.0) Gecko/20100101 Firefox/17.0).

我也可以在wireshark中看到这些相同的调用...

这就是为什么我问这是否是正常行为,如果响应没有返回,则尝试重新发送请求,类似于 TCP 握手和SYN 重新传输。

新更新

这似乎与我的 Ajax 调用无关。如果我用一个简单的 HREF 创建一个按钮。

<a href="/importdevice?device=serverA" class="btn btn-success">TestDirect</a>

然后在我的“实时 HTTP 标头输出”中,我得到...只有一个实例。

#request# GET http://172.16.118.15/importdevice?device=serverA

但我又一次在我的服务器日志中得到了。

[Sun Jan 13 03:20:25 2013] [debug] GET /importdevice (Mozilla/5.0 (Macintosh; Intel Mac OS  X 10.8; rv:17.0) Gecko/20100101 Firefox/17.0).
[Sun Jan 13 03:20:48 2013] [debug] GET /importdevice (Mozilla/5.0 (Macintosh; Intel Mac OS X 10.8; rv:17.0) Gecko/20100101 Firefox/17.0).

在服务器上的 wireshark 中,我也看到了两次......正如预期的那样,我的服务器代码也被调用了两次......这真的让我感到困惑。

【问题讨论】:

  • 这显然不是你真正的电话,因为它包含语法错误。请使用复制粘贴。
  • retryLimit 属性在那里做什么?这是所有相关的代码吗?
  • 好的,我已经准确地剪切并粘贴了我的代码。这没有任何错误,我的网络框架工作扩展了 URL。但我已经用直接网址对其进行了测试,我也遇到了同样的问题。
  • 抱歉,'retryLimit' 不相关。

标签: jquery html ajax browser request


【解决方案1】:

查看这篇解释正在发生的事情的博客文章: http://geek.starbean.net/?p=393

根据HTTP/1.1 RFC 8.2.4

如果 HTTP/1.1 客户端发送的请求包含请求正文,但不包含期望为“100-continue”的 Expect 请求头字段,并且客户端未直接连接到 HTTP/ 1.1 源服务器,如果客户端在收到服务器的任何状态之前看到连接关闭,客户端应该重试请求。

确保您设计的网络应用能够容忍这些额外的请求。而如果你想让 ajax 请求只做一次,比如向数据库写入一些东西,那么你需要将请求设计为幂等的。见:What is an idempotent operation?

此外,这突出了 GET 和 POST 操作之间的区别: http://www.cs.tut.fi/~jkorpela/forms/methods.html

作为一般设计实践:
-GET 操作应设计为幂等
-POST 操作应该用于写入数据库等。

【讨论】:

  • 谢谢!这正是我要找的……还有更多!
  • 因此按照HTTP规范,即使是非幂等请求,即POST也可以被客户端重试
  • @Yadu 是的,即使是 POST 请求也可以被客户端重试(正如提问者所经历的那样)。但是将 all POST 请求称为非幂等并不准确。在某些情况下,可能希望将 POST 请求设计为幂等的。请务必阅读this page 底部的最后一部分;标题为“使用“POST”进行幂等查询的可能原因”的部分
  • 所以,这被标记为已回答,我猜是这样,但实际上没有人说我是否可以做任何事情,除了在我的服务器端做很多工作来处理重复的请求。有没有办法告诉浏览器不要重新发送 POST 请求或禁用它们的超时 - 最好是从 javascript/ajax?
  • 我现在也有类似的问题。假设第一个请求到达服务器,服务器需要 3 分钟来处理。 2分钟后,第二个请求进来了。我的服务器会立即返回400。再过 1 分钟,服务器将返回 200(第一个请求完成)。那是对的吗?现在的问题是客户端在收到 400 响应时显示错误。它不会等待 200 响应。
【解决方案2】:

如果您的请求异步,为什么要设置回调?

如果您想要一个同步过程,请将 beforeSend 代码放在 调用之前,并将 success 回调放在调用之后。

如果您想要异步方法,请设置 async: true

这行代码也是错误的:

url: '<%= url_for('importDevice') %>',

看引号:应该是

url: '<%= url_for("importDevice") %>',

也许这些建议不会解决问题,但会为您指明正确的方向。问题可能出在发布的代码之外的其他地方...

回到你的问题: 会不会是浏览器?我不认为,但谁知道那里有什么!......浏览器是什么?你试过另一个吗?

我建议尝试使用 Firefox 并使用 Live httpHeaders 插件(非常易于设置和使用)。您将看到从客户端输出的内容,而不是看到从服务器端输入的内容。

【讨论】:

  • 谢谢 Paolo,是的,我想关闭异步。我的浏览器是 chrome,我也尝试过 safari 并看到相同的行为。在开发工具下的 chrome 中,我看到一个状态为“待处理”的 POST。但是,在我的服务器日志和wireshark 中,我看到POST 请求通过了两次。谢谢,我会安装 Live httpHeaders 插件,看看是否有帮助
  • 以上更新。它似乎与 Ajax 无关......现在我真的迷路了。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2015-01-28
  • 2016-04-24
  • 1970-01-01
  • 2019-09-11
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多