web超时配置总结

欢迎光临我的博客拓跋的前端客栈,这个是原文地址。如果您发现我文章中存在错误,请尽情向我吐槽,大家一起学习一起进步φ(>ω<*)

引子


前段时间做项目时,由于某个请求在后台要进行长时间的脚本处理,频频超出系统默认的超时时长,于是我就对超时时长进行了检查。由于超时的时候请求报错的提示信息并不明确,为了修改这玩意儿也查了一堆资料,忙活了半天,终于把超时问题解决了,在这里做个记录。

我的项目用的是web前台(这个无所谓用的什么框架,反正归根结底是js)+nginx代理+Node.js后台搭建的,我发现每一节都设了一道timeout超时的关卡,也就是说有客户端超时代理超时服务端超时三种情况。下面针对每种超时,我将进行分别讲解。

timeout.jpg

客户端超时


XMLHttpRequest的超时设置

讲客户端超时,就要从XMLHttpRequest来谈起。

XMLHttpRequest是一个API,它为客户端提供了在客户端和服务器之间传输数据的功能。它提供了一个通过URL来获取数据的简单方式,并且不会使整个页面刷新。这使得网页只更新一部分页面而不会打扰到用户。XMLHttpRequest在AJAX中被大量使用。

XMLHttpRequest一开始只是微软浏览器提供的一个接口,后来各大浏览器纷纷效仿也提供了这个接口,再后来W3C对它进行了标准化,提出了XMLHttpRequest标准。XMLHttpRequest标准又分为Level 1和Level 2。
XMLHttpRequest Level 1主要存在以下缺点:

  • 受同源策略的限制,不能发送跨域请求;
  • 不能发送二进制文件(如图片、视频、音频等),只能发送纯文本数据;
  • 在发送和获取数据的过程中,无法实时获取进度信息,只能判断是否完成;

那么Level 2对Level 1 进行了改进,XMLHttpRequest Level 2中新增了以下功能:

  • 可以发送跨域请求,在服务端允许的情况下;
  • 支持发送和接收二进制数据;
  • 新增formData对象,支持发送表单数据;
  • 发送和获取数据时,可以获取进度信息;
  • 可以设置请求的超时时间;

在Level 2版本的XMLHttpRequest对象,增加了timeout属性,可以设置HTTP请求的时限。

    xhr.timeout = 60*1000;
    xhr.ontimeout = function(event){
    alert('timeout!');
  }

在上述代码中,请求超时时间被设为1分钟(xhr.timeout的单位是毫秒),如果超过1分钟的时限,则会自动停止http请求;同时,触发ontimeout事件,弹出“timeout!”的提示框。

同时值得注意的一点是,超时时间的计算,是从调用xhr.send()开始,至xhr.loadend触发为止的这段时间。即时xhr.timeout的设置是在xhr.send()之后,timeout的计时起点仍为调用xhr.send()的时刻。

其他超时设置

其实会了XMLHttpRequest的超时设置,其他前端的框架啊、工具啊的超时设置都不再是问题,这就有点万法归宗的意思。因为我们常用的jQuery.ajax()方法实际上就是对浏览器提供的XMLHttpRequest对象的封装。而又有很多其他框架或者工具的请求模块是对jQuery.ajax()的封装。说到底,都是依赖的XMLHttpRequest对象。因此掌握了XMLHttpRequest,其他都很好学会。

依jQuery.ajax()的超时设置为例:

    $.ajax({
        url: "test.html",
        error: function(){
            // will fire when timeout is reached
        },
        success: function(){
            // do something
        },
        timeout: 60*1000 // sets timeout to 1 minute
    });

实在是简单,对不对?

代理超时


nginx的超时设置主要分3种:

  • proxy_connect_timeout
  • proxy_read_timeout
  • proxy_send_timeout

proxy_connect_timeout

语法: proxy_connect_timeout timeout_in_seconds
上下文: http, server, location
默认值: 60s

proxy_connect_timeout是和后端建立连接的超时时间。需要记住的是,这个时间不能超过75秒。

这个不是等待后端返回页面的时间,那是由proxy_read_timeout声明的。如果你的upstream服务器起来了,但是挂起了(例如,没有足够的线程处理请求,所以把你的请求放到请求池里稍后处理),那么这个声明是没有用的,由于与upstream服务器的连接已经建立了。

proxy_read_timeout

语法: proxy_read_timeout the_time
上下文: http, server, location
默认值: 60s

proxy_read_timeout是从后端读取数据的超时时间,两次读取操作的时间间隔如果大于这个值,和后端的连接会被关闭。如果一个请求时间时间非常大,要把这个值设大点。

proxy_send_timeout

语法: proxy_send_timeout time
上下文: http, server, location
默认值: 60s

proxy_send_timeout是向后端写数据的超时时间,两次写操作的时间间隔大于这个值,也就是过了这么长时间后端还是没有收到数据,连接会被关闭。

服务端超时


Node.js做服务器时,默认的超时时长为2分钟。假设请求发送到服务端,在2分钟内没有响应返回给客户端,客户端的链接就会被重置。这个时长的设置是很必要的,因为过长的请求响应会阻塞IO,造成极差的用户体验。特别对于Node.js这种单线程应用来说,影响可以说是灾难性的,因此如何设置服务端的超时时间也是非常关键的。

服务端超时设置也不是一刀切的,总体来说,有一个总开关,还可以对每个服务端的请求或者响应单独设置超时时长。

我们后面的代码均以Node.js+Express搭建的后台为例。

服务端超时总开关

使用Node.js的http模块启动服务器,并设置超时时长:

    const express = require('express');
    const config = require(./config);

    process.env.PORT = config.webport || 8888;

    let app = express();
    let server = app.listen(process.env.PORT);
    server.setTimeout(60*1000);
    server.on('timeout', function(){
        console.log('timeout!')
    })

server.setTimeout()是设置server的超时时间设置方法,单位是毫秒,一旦超过设置的超时时长,则触发server对象的'timeout'事件,并传入socket作为一个参数。

设置Express中间件超时时间

Express是一个路由和中间件Web框架,其自身只具有最低程度的功能:Express应用程序基本上是一系列中间件函数调用。

中间件函数能够访问请求对象 (req)、响应对象 (res) 以及应用程序的请求/响应循环中的下一个中间件函数。下一个中间件函数通常由名为 next 的变量来表示。

中间件函数可以执行以下任务:

  • 执行任何代码。
  • 对请求和响应对象进行更改。
  • 结束请求/响应循环。
  • 调用堆栈中的下一个中间件函数。

如果当前中间件函数没有结束请求/响应循环,那么它必须调用next(),以将控制权传递给下一个中间件函数。否则,请求将保持挂起状态。

对于Express的中间件,我们可以在app.js里为每个路由及其处理程序函数的中间件设置请求的响应和超时时间,

    app.get('/user/:id', function (req, res, next) {
        // 设置该中间件下所有HTTP请求的超时时间
        req.setTimeout(60*1000);
        // 设置该中间件下所有HTTP请求的服务器响应超时时间
        res.setTimeout(60*1000);
        res.send('USER');
    });

Express不仅可以针对中间件设置超时时间,还可以针对某个具体的接口设置请求和响应的超时时长:

    router.get('/user/:id', function (req, res, next) {
        // 设置该接口下所有HTTP请求的超时时间
        req.setTimeout(60*1000);
        // 设置该接口下所有HTTP请求的服务器响应超时时间
        res.setTimeout(60*1000);
        res.send('USER');
    });

小结

全写完了以后,仔细想了想,发现还没有写全,比如说服务端也可以发请求,不管是用http.method还是用request模块,都可以发请求。由于Node.js在很多项目中都是作为中间层,不管是向JAVA后台请求数据,还是用作解决客户端跨域问题的请求中转层,发请求都是很常用的用法。但是只要读者能耐心读到这里,相信随手网上搜一下就知道怎么写了,因此我这里就不再赘述了。

写这篇文章主要想分享一下我的感受,超时限制就像一道关卡,一个请求从客户端到服务端,再从服务端返回客户端的路上,要“过五关斩六将”。有的时候想要修改超时设置,可能不是简单修改某一处就能解决问题的,这时候一定要仔细检查,把这个系统理一遍,每一个地方的超时设置都看一看,才能解决问题~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270

推荐阅读更多精彩内容

  • Ajax和XMLHttpRequest 我们通常将Ajax等同于XMLHttpRequest,但细究起来它们两个是...
    changxiaonan阅读 2,149评论 0 2
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,100评论 18 139
  • 看到标题时,有些同学可能会想:“我已经用xhr成功地发过很多个Ajax请求了,对它的基本操作已经算挺熟练了。” 我...
    前端渣渣阅读 5,705评论 1 12
  • 本文详细介绍了 XMLHttpRequest 相关知识,涉及内容: AJAX、XMLHTTP、XMLHttpReq...
    semlinker阅读 13,514评论 2 18
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,507评论 6 13