- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在开发一个将 http 请求(GET 用于测试目的)发送到 java servlet 的设置。它的工作原理是,ser let 从浏览器获取请求,解析它并通过 TCP 套接字将其发送到“主”服务器,后者处理请求并发回响应。然后 servlet 提取先前存储在 ConcurrentHashMap 中的 HttpServletResponse,打开 PrintWriter,并发回响应。一切都进行得很顺利,除了 HttpServletResponse 并不总是发回写入 PrintWriter 的信息。浏览器每次都会收到“OK”响应,但响应通常不包含我尝试编写的任何信息。
下面我有用于传递 HttpServletResponse 实例的初始 doGet 的代码,然后是写入响应缓冲区的方法。之后包括浏览器收到的响应。之后,一些关于如何可靠地获得预期结果的观察,以防有助于确定问题。
请注意,唯一的变量似乎是响应是否被写入;我查看了输出日志,但在按预期写入响应的时间与未按预期写入响应的时间之间找不到任何其他差异。我写的HTTPServletResponseListener每次都会收到响应。
[使用 Glassfish 3.1.1]
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String js = request.getParameter("json");
// Omitting try-catch for space
Message msg = this.parser.parseToMessage(js);
this.sc.send(msg, new HTTPServletResponseListener(response));
}
以及响应时调用的 HTTPServletResponseListener 方法(除了仅将本地 HttpServletResponse 分配给本地字段的构造函数之外,这是唯一的方法)
public void handleResponse(ResponseMessage response) {
DataParser parser = new JSONParser();
String temp = parser.parseToString(response);
httpResponse.setContentType("application/json");
httpResponse.addHeader("Hmm","yup");
try {
PrintWriter out = httpResponse.getWriter();
out.println(temp);
} catch (IOException ex) {
Logger.getLogger(HTTPServletReponseListener.class.getName()).log(Level.SEVERE,null,ex);
}
}
响应和浏览器接收它们:
当它按预期工作时:
当响应为空时:
HTTP/1.1 200 OK
X-Powered-By: Servlet/3.0 JSP/2.2 (GlassFish Server Open Source Edition 3.1.1 Java/Sun Microsystems Inc./1.6)
Server: GlassFish Server Open Source Edition 3.1.1
Content-Length: 0
Date: Thu, 16 Feb 2012 15:26:35 GMT
当响应符合预期时:
HTTP/1.1 200 OK
Hmm: yup
X-Powered-By: Servlet/3.0 JSP/2.2 (GlassFish Server Open Source Edition 3.1.1 Java/Sun Microsystems Inc./1.6)
Server: GlassFish Server Open Source Edition 3.1.1
Content-Type: application/json;charset=ISO-8859-1
Content-Length: 126
Date: Thu, 16 Feb 2012 15:27:30 GMT
观察:
按照这些步骤,我可以在 95% 的时间内获得响应......否则,它的成功率约为 50%。
在浏览器上点击刷新...部署后的前几个请求往往会更频繁地工作等待至少 5 秒,然后再次刷新(发送另一个测试请求)这往往可靠地工作。如果失败,您必须等待 15 秒,然后它会再次起作用。
如果在等待响应之前写入 HttpServletResponse,则它每次都有效。
非常感谢您花时间阅读本文。我读过其他 Stackoverflow 问题,但似乎没有涉及到这个特定问题,除非我错过了连接。
最佳答案
这是最有趣的一行:
this.sc.send(msg, new HTTPServletResponseListener(response));
我怀疑 sc
是您正在调用的外部 TCP 服务器,并且您还传递了一个监听器,以便在响应到达时收到通知。现在重要的假设是:我认为 TCP 服务器异步发送响应是否正确,在不同的线程中通知您的监听器?
如果是这种情况,则可以解释您的行为。在 3.0 之前的 servlet 中,您必须在 doGet
中处理整个请求。一旦您的代码离开 doGet()
,servlet 容器会假定整个请求已被处理并丢弃该请求。
您引入了一个竞争条件:doGet()
返回时没有向输出流写入任何内容。容器处理响应并将其发回需要几毫秒。如果在这段短时间内您的外部 TCP 服务器返回数据并通知监听器,则数据将通过。但是,如果服务器速度稍慢,则说明您正在向已处理的连接发送响应。
这样想:浏览器进行调用,doGet()
被调用,后者又调用后端服务器。 doGet()
返回并且 servlet 容器假定您已完成。它发回(空)响应并忘记此请求。几毫秒甚至几秒后响应从后端服务器返回。但是连接消失了,它已经被发送,套接字被关闭,浏览器呈现响应。你不是在告诉你的容器:嘿,等等,我还没有完成那个响应!。
从最坏到最好:
在 doGet()
中主动等待/轮询响应。
使您的外部 TCP 服务器调用阻塞。更改 TCP 服务器外观,使其返回 ResponseMessage
并将监听器代码移至doGet()
。示例:
ResponseMessage responseMsg = this.sc.send(msg);
DataParser parser = new JSONParser();
String temp = parser.parseToString(responseMsg);
httpResponse.setContentType("application/json");
httpResponse.addHeader("Hmm","yup");
PrintWriter out = httpResponse.getWriter();
out.println(temp);
使用 Servlet 3.0 异步支持,这在您的用例中是一个更好的选择,并且更改的范围将非常有限。
在 doGet()
中:
final AsyncContext asyncContext = request.startAsync(request, response);
完成后在 HTTPServletResponseListener
中:
asyncContext.complete();
对 startAsync()
的额外调用告诉容器:即使我已从 doGet()
返回,我还没有完成此请求。请等一下。我写了一个article前段时间关于Servlet 3.0的。
关于java - HttpServletResponse 似乎定期过早发送,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9316904/
我是 .NET 编程的新手,目前我正在开发一个计算机健康监控系统,该系统目前处于初级阶段。我将使用 C# 2010 并使用 WMI 查询来查询计算机信息。 在进一步开发应用程序之前,我创建了一个迷你测
我正在考虑开发一个站点,服务器将定期抓取另一个站点,以便为我的数据库中的某些条目收集内容。我的问题如下... 如何让服务器及时执行抓取? 你能让它执行 php 还是你用什么语言来执行抓取? 有什么好的
使用 JobScheduler , 我设置了一个简单的 JobService如下: @TargetApi(21) public class SimpleJobService extends JobSe
我想定期从我的 JavaFX 应用程序执行任务。该任务从远程流中提取数据。 虽然我知道我可以按照下面的建议使用计时器: JavaFX periodic background task 我相信这应该能够
上下文 我们使用多个具有相同工作流程的 git 存储库,涉及两个分支,并且想知道如何最好地将一个提交“同步”到另一个。 简单地说,我们的 git 存储库包含: 长寿枝 两个分支: master(持续开
我正在编写一个需要以高精度和低频率进行后台位置更新的应用程序。解决方案似乎是一个后台 NSTimer 任务,它启动位置管理器的更新,然后立即关闭。之前有人问过这个问题: How do I get a
我有一个包含产品的商业系统。产品被分类并且还具有属性组。 我想同步/更新将来自商务系统的产品转换为在 Orchard 中定义的产品项目的内容类型,以便将数据从商务系统推送到 Orchard 内容项目。
所以我想做的只是按时间间隔更新 Activity 的 View ,比如每秒一次。在这个特定的例子中有几个按钮,我想要改变的只是它们的文本。我在这里阅读了很多解决相同问题的问题,但我似乎比其他人更容易陷
我有一个项目的 recyclerView,(除其他外)包含从 WS 检索的游戏持续时间的时间戳。为了显示比赛的持续时间,我必须获取当前时间戳,做垫子并将其转换为可读格式(例如:5 分钟 5 秒)。只有
这可能是一个愚蠢的问题,但经过几个小时的谷歌搜索后我找不到合适的答案.. 我们有一个在 cloudbees 上运行的业务关键型应用程序。源代码已正确备份,我们希望我们的数据库也有相同的备份。 Clou
我一直在观察我的 PostgreSQL 只读副本显示复制滞后的周期性延迟。延迟似乎会增加到 30-40 分钟,然后自动下降到 0。这与 CPU 利用率 相关,但它远未接近 CPU 限制。 读取流量来自
我已完成 Keepalived + MySQL(master - master)设置。 我将 MASTER 和 BACKUP 的优先级保持不变,因为我不希望它们开始频繁抖动(一次切换 VIP 就足够了
我目前有一个图像,如果本地内容可用,它会被 ping 以仅显示一个 div。 但我认为更好的实现方式是以 30 秒为间隔定期检查 ping,如果图像未成功 ping,则显示 #offline div。
我正在尝试编写一项服务,该服务将每隔 X 分钟尝试获取设备的 GPS 位置,并在后台运行和记录,即使应用程序未处于焦点状态也是如此。 那么,是时候创建服务了。 我创建了一个服务,设置了 locatio
我在这里阅读了很多问题,但无法弄清楚问题是什么。 我正在为 Android 编写现场服务应用程序。在其中一个 Activity (MyActivity.java) 中,我有两个按钮:开始和停止。 当现
假设我有这样的代码: git fetch && git diff origin origin/master --quiet || echo "untracked" 我希望它定期运行。比方说,每秒一次
我想建立与 HTTPS 服务器(例如 google.com)的连接并定期获取新鲜内容。 我编写了简单的 HTTP 客户端: public class AsyncLoader { private
我想定期进行 wifi 扫描(可能每 5 分钟一次)并将结果导出到一个 txt 文件,以便它显示 AP SSID 和 MAC 以及接收到的信号强度。我是 Android 开发的新手,所以我只知道基础知
当手机处于 sleep 模式时,Workmanager 不会向我的服务器发送坐标。异步任务被杀死? 我使用 Workmanager (PeriodicWorkRequest) 每 15 分钟获取一次坐
目标 收集 LogCat 的定期更新并将这些文本 block 保存(附加)到 SD 卡上的文件中 问题 Log 类不提供自特定时间戳以来的更新 可能的解决方案 我的计划是定期运行类似于以下代码:htt
我是一名优秀的程序员,十分优秀!