java 轮询http_HTTP轮询模型

article/2025/10/14 9:36:07

HTTP轮询模型

长短轮询

http协议是一种client-server模型的应用层协议,这种c-s的模式虽然大多数情况都能满足需求,但是某些场景也需要服务端能够将一些信息实时的推送到客户端,即实现服务器向客户端推消息的功能。

比如:

配置管理中心服务端需要将更新的配置推送到client端

消息推送服务器需要将一些消息推送到Android、iOS客户端

利用Http协议实现服务器推送有两种常见的思路:

短轮询拉

客户端不停的去向服务器发送轮询请求,如果有数据更新,客户端能也能尽快(取决于轮询间隔)获取最新的数据,这种方式被称为Http短轮询。

3eda032f9285b9963b1861aaf0763d69.png

这种方式有如下缺点:

因为是短轮询,因此一定时间t内需要进行轮询的次数就更多,而Http的连接是需要tcp三次握手等资源开销的。

由图可以看出,但是服务端数据发生更新时,客户端并不是立刻收到更新的数据(除去网络传输仍然还需要时间),而只能是在下一次轮询的时候才能感知到数据的变更。

长轮询推

长轮询的思路是这样的:尽量减少轮询的次数,从而减少资源开销。为了减少轮询次数,那么每次轮询的时间跨度就需要比较长,因此成为长轮询,同时也希望长轮询模型的每一次轮询效率要高于短轮询。

9ad8718d972c910320bbd933c63443f7.png

长轮询模型有这么几个特征:

每次轮询的间隔不固定

服务器对每次轮询做出响应的条件是:超时或者数据更新

长轮询模型中,客户端能实时感知到服务器端数据更新

由于很多服务器都具有异步处理连接的能力,因此图中的阻塞消耗的资源比较小。

异步Servlet

下面是利用Servlet规范中提供的异步Servlet作为服务端的Http长轮询模型,实现了客户端能实时获取服务端某个配置文件内容。

异步Servlet是Servlet3.0出来的新特性,对于需要异步处理的连接,Servlet引擎会将处理该请求的工作线程回收进工作线程池,而不是阻塞在该请求上。

package httplongconnection;

import java.io.File;

import java.io.FileFilter;

import java.io.FileInputStream;

import java.io.FileNotFoundException;

import java.io.IOException;

import java.io.InputStream;

import java.io.PrintWriter;

import javax.servlet.AsyncContext;

import javax.servlet.AsyncEvent;

import javax.servlet.AsyncListener;

import javax.servlet.ServletException;

import javax.servlet.annotation.WebServlet;

import javax.servlet.http.HttpServlet;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import org.apache.commons.codec.digest.DigestUtils;

import org.apache.commons.io.monitor.FileAlterationListener;

import org.apache.commons.io.monitor.FileAlterationMonitor;

import org.apache.commons.io.monitor.FileAlterationObserver;

@WebServlet(urlPatterns = "/long", asyncSupported = true)

public class HttpLongConnectionServlet extends HttpServlet {

/**

*

*/

private static final long serialVersionUID = 1L;

static FileAlterationObserver observer;

static {

FileAlterationMonitor monitor = new FileAlterationMonitor(1000L);// 每隔1000毫秒扫描一次

// 需要监听的文件目录

observer = new FileAlterationObserver(new File("E:/J2EE_workspace/httplongconnection/src/main/resources"), new FileFilter() {

public boolean accept(File pathname) {

// TODO Auto-generated method stub

return true;

}

});

System.out.println("observer");

monitor.addObserver(observer);

try {

monitor.start();

} catch (Exception e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

@Override

protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {

// TODO Auto-generated method stub

final AsyncContext ctx = req.startAsync();

ctx.addListener(new AsyncListener() {

public void onTimeout(AsyncEvent event) throws IOException {

// TODO Auto-generated method stub

ctx.complete();

observer.removeListener((FileAlterationListener)ctx.getRequest().getAttribute("fileListener"));

}

public void onStartAsync(AsyncEvent event) throws IOException {

// TODO Auto-generated method stub

}

public void onError(AsyncEvent event) throws IOException {

// TODO Auto-generated method stub

}

public void onComplete(AsyncEvent event) throws IOException {

// TODO Auto-generated method stub

observer.removeListener((FileAlterationListener)ctx.getRequest().getAttribute("fuck"));

}

});

ctx.setTimeout(50 * 1000);

new Thread(new BizProcessor(ctx)).start();

}

class BizProcessor implements Runnable {

private String checkSum;

private AsyncContext asyncContext;

private boolean checkSumEqual(InputStream is, String originalCheckSum) {

try {

String digest = DigestUtils.md5Hex(is);

if (digest.equals(originalCheckSum)) {

return true;

}

this.checkSum = digest;

System.out.println(

"File has changed. new md5 is " + this.checkSum + ", old checsum is " + originalCheckSum);

return false;

} catch (FileNotFoundException e) {

// TODO Auto-generated catch block

e.printStackTrace();

} catch (IOException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

return false;

}

public BizProcessor(AsyncContext asyncContext) {

super();

this.asyncContext = asyncContext;

}

public void run() {

// TODO Auto-generated method stub

// sleep

HttpServletRequest req = (HttpServletRequest) asyncContext.getRequest();

String cSum = String.valueOf(req.getParameter("checkSum"));

// 文件update

InputStream is = null;

try {

is = new FileInputStream(new File("E:/J2EE_workspace/httplongconnection/src/main/resources/config.txt"));

} catch (FileNotFoundException e1) {

// TODO Auto-generated catch block

e1.printStackTrace();

}

if (is != null && !checkSumEqual(is, cSum)) {

try {

is.close();

is = this.getClass().getClassLoader().getResourceAsStream("config.txt");

PrintWriter out = asyncContext.getResponse().getWriter();

String content = org.apache.commons.io.IOUtils.toString(is, "UTF-8");

System.out.println(content);

out.write(checkSum + "\002" + content);

out.flush();

out.close();

is.close();

} catch (IOException e) {

e.printStackTrace();

}

asyncContext.complete();

}

System.out.println("register");

register(asyncContext);

// 没有发生文件更新,则等待超时发生

}

private void register(AsyncContext ctx) {

// TODO Auto-generated method stub

FileListerAdapter listner = new FileListerAdapter(ctx);

ctx.getRequest().setAttribute("fileListener", listner);

observer.addListener(listner);

}

}

}

Hello World!

poll();

function poll() {

$.ajax({

url: "/httplongconnection/long",

data : {"checkSum" : cSum},

success: function(response) {

if (response == null || response.length == 0) {

poll();

return;

}

var msg = response.split("\002");

var checkSum = md5(msg[1]), content = msg[1];

if (checkSum != cSum) {

cSum = checkSum;

$("#show").val(content);

}

poll();

}});

}

package httplongconnection;

import java.io.File;

import java.io.FileInputStream;

import java.io.IOException;

import java.io.InputStream;

import java.io.PrintWriter;

import javax.servlet.AsyncContext;

import org.apache.commons.codec.digest.DigestUtils;

import org.apache.commons.io.monitor.FileAlterationListenerAdaptor;

public class FileListerAdapter extends FileAlterationListenerAdaptor {

AsyncContext ctx;

public FileListerAdapter(AsyncContext ctx) {

super();

this.ctx = ctx;

}

@Override

public void onFileChange(File file) {

if (!file.exists() || !file.canRead()) {

System.out.println("The file " + file + " is not exists or is not readable!");

return;

}

try {

InputStream is = new FileInputStream(file);

if (ctx.getResponse().isCommitted()) {

return;

}

PrintWriter out = ctx.getResponse().getWriter();

String content = org.apache.commons.io.IOUtils.toString(is, "UTF-8");

System.out.println(content);

String digest = DigestUtils.md5Hex(is);

out.write(digest + "\002" + content);

System.out.println("yy");

System.out.println(content);

is.close();

out.flush();

out.close();

} catch (IOException e) {

e.printStackTrace();

}

ctx.complete();

//TODO 读取操作

super.onFileChange(file);

}

@Override

public void onFileCreate(File file) {

//TODO 读取操作

super.onFileCreate(file);

}

@Override

public void onFileDelete(File file) {

super.onFileDelete(file);

}

@Override

public void onDirectoryChange(File directory) {

System.out.println("----The directory " + directory + " has changed.");

super.onDirectoryChange(directory);

}

@Override

public void onDirectoryCreate(File directory) {

super.onDirectoryCreate(directory);

}

@Override

public void onDirectoryDelete(File directory) {

super.onDirectoryDelete(directory);

}

}


http://chatgpt.dhexx.cn/article/XA0wZGLZ.shtml

相关文章

七种轮询介绍(后附实践链接)

我有一个朋友~ 做了一个小破站,现在要实现一个站内信web消息推送的功能,对,就是下图这个小红点,一个很常用的功能。 不过他还没想好用什么方式做,这里我帮他整理了一下几种方案,并简单做了实现…

linux cgroup 死循环,Linux CGroup 基础

CGroup V1 1. CGroup 概念Task: 任务,也就是进程,但这里的进程和我们通常意义上的 OS 进程有些区别,在后面会提到。 CGroup: 控制组,一个 CGroup 就是一组按照某种标准划分的Tasks。这里的标准就是 Subsystem 配置。换句话说&…

linux cgroup 原理,Cgroup框架的实现

CGoup核心主要创建一系列sysfs文件,用户空间可以通过这些节点控制CGroup各子系统行为,以及各子系统模块根据参数。在执行过程中或调度进程到不同CPU上,或控制CPU占用时间,或控制IO带宽等等。另外,在每个系统的proc文件…

CGroup的原理和使用

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、主要功能二、基本概念三、cgroups子系统介绍四、cgroups 层级结构五、数据结构 前言 Linux CGroup全称Linux Control Group, 是Linux内核的一个…

Linux cgroup介绍

本文参考网上一些资料,结合实际应用,简要介绍一下cgroup。 为什么要有cgroup Linux系统中经常有个需求就是希望能限制某个或者某些进程的分配资源。也就是能完成一组容器的概念,在这个容器中,有分配好的特定比例的cpu时间&#…

漫谈cgroup

什么是cgroup cgroup 是linux内核的一个功能,用来限制、控制与分离一个进程组的资源(如CPU、内存、磁盘I/O等)。它是由 Google 的两位工程师进行开发的,自 2008 年 1 月正式发布的 Linux 内核 v2.6.24 开始提供此能力。 cgroup …

容器中的Cgroup

文章目录 容器中的CgroupCgroup概念容器化两个关键核心现代容器化带来的优势什么是Cgroup Cgroup的一些测试测试CPU和内存使用情况CPU 周期限制 CPU Core 控制CPU 配额控制参数的混合使用内存限额Block IO 的限制bps 和 iops的限制 容器中的Cgroup Cgroup概念 容器化两个关键…

Cgroup 资源配置

目录 一、Cgroup定义 二、使用stress压力测试工具测试cpu和内存状态 1、创建一个dockerfile文件 2、创建镜像 3、创建容器 ①创建容器 ②、创建容器并产生10个子函数进程 三、CPU 周期 1、实现方案 2、实验 四、CPU Core控制 五、docker build 一、Cgroup定义 cg…

cgroup资源配置

一、cgroup介绍二、利用stress 压力测试工具来测试三、CPU控制1、仅用率控制(权重)2、周期限制方法一:在命令行里直接设置方法二:创建容器后,关闭容器在文件里直接修改方法三:进入容器查看 3、cpu核心数4、…

【CGroup原理篇】3. CGroup使用指南

写在前面 这里先从整体上概述cgroup的创建,挂载,参数配置和卸载,后面的章节中会一一介绍每个子系统的详细使用方法和使用案例。 一、使用Linux命令管理CGroup 1.1挂载cgroup临时文件系统 mount -t tmpfs cgroup_root /sys/fs/cgroup 1.2 创建挂载层级需要的目录 mkdir /sy…

Linux CGroup 原理

Linux CGroup 原理 1、CGroup简介 cgroups是Linux下控制一个(或一组)进程的资源限制机制,全称是control groups,可以对cpu、内存等资源做精细化控制。 开发者可以直接基于cgroups来进行进程资源控制,比如8核的机器上…

LINUX CGROUP总结

简介: Linux CGroup全称Linux Control Group, 是Linux内核的一个功能,用来限制,控制与分离一个进程组群的资源(如CPU、内存、磁盘输入输出等)。这个项目最早是由Google的工程师在2006年发起(主要是Paul Men…

cgroup 简介

cgroup 的功能在于将一台计算机上的资源(CPU,memory,network)进行分片,来防止进程间不利的资源抢占。 术语 cgroup:关联一组 task 和一组 subsystem 的配置参数。 一个 task 对应一个进程,cg…

cgroup基础介绍

一项新概念的产生,必然有其原因,cgroup也不例外,最初由谷歌工程师Paul Menage和Rohit Seth提出【1】:因为计算机硬件能力越来越强大,为了提高机器的使用效率,可以在同一台机器上运行不同运算模型的工作。开…

深入浅出cgroup

一、什么是cgroup Cgroup是linux内核用来控制系统资源的机制,它将操作系统中的所有进程以组为单位划分,给这一组进程定义对某一类资源特定的访问权限。Cgroup用子系统(subsystem)来描述所能控制的系统资源,子系统具有多…

Cgroup概述

一、Cgroup的目的 Cgroup和namespa类似,也是将进程进程分组,但是目的与namespace不一样,namespace是为了隔离进程组之前的资源,而Cgroup是为了对一组进程进行统一的资源监控和限制。 二、为什么需要Cgroup 在Linux里&#xf…

Cgroup是什么(相关概念、功能、作用、特点、怎么用)

文章目录 Cgroup 什么是CgroupCgroup的相关概念相互关系Cgroup的功能Cgroup的作用Cgroup的层级图及特点Cgroup怎么用 什么是Cgroup cgroups,其名称源自控制组群(control groups)的简写,是Linux内核的一个功能,用来限…

Cgroup简介-概述

Cgroup(Control Groups)是这样一种机制:它以分组的形式对进程使用系统资源的行为进行管理和控制。也就是说,用户通过cgroup对所有进程进行分组,再对该分组整体进行资源的分配和控制。 1 Cgroup的结构 cgroup中的每个分…

J2EE技术架构

一、简介 J2EE(Java 2 Platform, Enterprise Edition)是一个为大企业主机级的计算类型而设计的Java平台。Sun微系统(与其工业伙伴一起,例如IBM)设计了J2EE,以此来简化在受客户级环境下的应用开发。由于创造…