jax-rs jax-ws

服务器发送的事件 (或简称为SSE )是非常有用的协议,它允许服务器通过HTTP将数据推送到客户端。 这是我们的Web浏览器支持的年龄,但是令人惊讶的是, JAX-RS规范在很长一段时间内都忽略了这一点。 尽管Jersey提供了适用于SSE媒体类型的扩展名,但该API尚未正式化,因此不能移植到其他JAX-RS实现中。

幸运的是, JAX-RS 2.1 (也称为JSR-370)通过将SSE支持(客户端和服务器端)作为正式规范的一部分,已经改变了这一点。 在今天的帖子中,我们将研究如何使用最近发布的出色的Apache CXF框架3.2.0版将SSE支持集成到现有的Java REST(ful) Web服务中。 实际上,除了自举之外,实际上没有CXF特定的东西,所有示例都应在实现JAX-RS 2.1规范的任何其他框架中工作。

事不宜迟,让我们开始吧。 由于这些天大量的Java项目都建立在出色的Spring Framework之上,因此我们的示例应用程序将使用Spring Boot和Apache CXF Spring Boot Integration使我们Swift起步。 老好伙伴Apache Maven也可以通过管理项目依赖项来帮助我们。

org.springframework.bootspring-boot-starter1.5.8.RELEASEorg.apache.cxfcxf-rt-frontend-jaxrs3.2.0org.apache.cxfcxf-spring-boot-starter-jaxrs3.2.0org.apache.cxfcxf-rt-rs-client3.2.0org.apache.cxfcxf-rt-rs-sse3.2.0

在后台, Apache CXF正在使用Atmosphere框架来实现SSE传输,因此这是我们必须包含的另一个依赖项。

org.atmosphereatmosphere-runtime2.4.14

有关依赖于Atmosphere框架的详细信息导致需要提供其他配置设置,即transportId ,以便确保在运行时拾取支持SSE的传输。 相关详细信息可以添加到application.yml文件中:

cxf:servlet:init:transportId: http://cxf.apache.org/transports/http/sse

太好了,所以基础就在那里,继续前进。 我们将要构建的REST(ful) Web服务将在SSE流中公开虚构的CPU平均负载(为简单起见,随机生成)。 Stats类将构成我们的数据模型。

public class Stats {private long timestamp;private int load;public Stats() {}public Stats(long timestamp, int load) {this.timestamp = timestamp;this.load = load;}// Getters and setters are omitted...
}

说到流, Reactive Streams规范已进入Java 9 ,希望我们能看到Java社区加速采用React式编程模型。 此外,在具有Reactive Streams支持的情况下,开发支持SSE的 REST(ful) Web服务将变得更加容易和直接。 为此,让我们将RxJava 2集成到示例应用程序中。

io.reactivex.rxjava2rxjava2.1.6

这是开始使用我们的StatsRestService类(典型的JAX-RS资源实现)的好时机。 JAX-RS 2.1中的关键SSE功能以Sse上下文对象为中心,可以像这样注入。

@Service
@Path("/api/stats")
public class StatsRestService {@Context public void setSse(Sse sse) {// Access Sse context here}

在Sse上下文之外,我们可以访问两个非常有用的抽象: SseBroadcaster和OutboundSseEvent.Builder ,例如:

private SseBroadcaster broadcaster;
private Builder builder;@Context
public void setSse(Sse sse) {this.broadcaster = sse.newBroadcaster();this.builder = sse.newEventBuilder();
}

您可能已经猜到了, OutboundSseEvent.Builder构造了OutboundSseEvent类的实例,这些实例可以通过电线发送,而SseBroadcaster则向所有连接的客户端广播相同的SSE流。 话虽如此,我们可以生成OutboundSseEvent的流并将其分发给感兴趣的每个人:

private static void subscribe(final SseBroadcaster broadcaster, final Builder builder) {Flowable.interval(1, TimeUnit.SECONDS).zipWith(eventsStream(builder), (id, bldr) -> createSseEvent(bldr, id)).subscribeOn(Schedulers.single()).subscribe(broadcaster::broadcast);
}private static Flowable<OutboundSseEvent.Builder> eventsStream(final Builder builder) {return Flowable.generate(emitter -> emitter.onNext(builder.name("stats")));
}

如果您不熟悉RxJava 2 ,请不用担心,这就是这里发生的情况。 eventsStream方法为stats类型的SSE事件返回有效的无限OutboundSseEvent.Builder实例流。 订阅方法稍微复杂一点。 我们首先创建一个流,该流每秒发出序号,例如fe 0、1、2、3、4、5、6等。 稍后,我们将此流与eventsStream方法返回的流组合在一起,实质上将两个流合并为一个流,该流每秒发出一个元组(number, OutboundSseEvent.Builder ) 。 公平地说,该元组对我们不是很有用,因此我们将其转换为OutboundSseEvent类的实例,将数字视为SSE事件标识符:

private static final Random RANDOM = new Random();private static OutboundSseEvent createSseEvent(OutboundSseEvent.Builder builder, long id) {return builder.id(Long.toString(id)).data(Stats.class, new Stats(new Date().getTime(), RANDOM.nextInt(100))).mediaType(MediaType.APPLICATION_JSON_TYPE).build();
}

OutboundSseEvent可以使用常规MessageBodyWriter解析策略在data属性中携带将相对于指定的mediaType进行序列化的任何有效负载。 一旦获得OutboundSseEvent实例,便使用SseBroadcaster :: broadcast方法将其发送出去。 请注意,我们使用subscribeOn运算符将控制流移交给了另一个线程,这通常是您一直在做的事情。

很好,希望现在清除了流部分,但是我们如何才能真正订阅SseBroadcaster发出的SSE事件? 这比您想像的要容易:

@GET
@Path("broadcast")
@Produces(MediaType.SERVER_SENT_EVENTS)
public void broadcast(@Context SseEventSink sink) {broadcaster.register(sink);
}

我们都准备好了。 这里最重要的是正在生成的内容类型,应将其设置为MediaType.SERVER_SENT_EVENTS 。 在这种情况下, SseEventSink的上下文实例变为可用,并且可以向SseBroadcaster实例注册。

要查看我们的JAX-RS资源,我们需要使用例如JAXRSServerFactoryBean引导服务器实例,并在此过程中配置所有必需的提供程序。 请注意,我们也在显式指定要使用的传输,在这种情况下为SseHttpTransportFactory.TRANSPORT_ID

@Configuration
@EnableWebMvc
public class AppConfig extends WebMvcConfigurerAdapter {@Beanpublic Server rsServer(Bus bus, StatsRestService service) {JAXRSServerFactoryBean endpoint = new JAXRSServerFactoryBean();endpoint.setBus(bus);endpoint.setAddress("/");endpoint.setServiceBean(service);endpoint.setTransportId(SseHttpTransportFactory.TRANSPORT_ID);endpoint.setProvider(new JacksonJsonProvider());return endpoint.create();}@Overridepublic void addResourceHandlers(ResourceHandlerRegistry registry) {registry.addResourceHandler("/static/**").addResourceLocations("classpath:/web-ui/"); }
}

要结束循环,我们只需要为Spring Boot应用程序提供运行器即可:

@SpringBootApplication
public class SseServerStarter {    public static void main(String[] args) {SpringApplication.run(SseServerStarter.class, args);}
}

现在,如果我们运行该应用程序并使用多个Web浏览器或同一浏览器中的不同选项卡导航到http:// localhost:8080 / static / broadcast.html ,我们将观察到所有事件内部绘制的相同事件流:

不错,广播当然是一个有效的用例,但是在每次端点调用时返回一个独立的SSE流又如何呢? 简单,只需使用SseEventSink方法(例如sendclose )即可直接操作SSE流。

@GET
@Path("sse")
@Produces(MediaType.SERVER_SENT_EVENTS)
public void stats(@Context SseEventSink sink) {Flowable.interval(1, TimeUnit.SECONDS).zipWith(eventsStream(builder), (id, bldr) -> createSseEvent(bldr, id)).subscribeOn(Schedulers.single()).subscribe(sink::send, ex -> {}, sink::close);
}

这次,如果我们运行该应用程序并使用多个Web浏览器或同一浏览器中的不同选项卡导航到http:// localhost:8080 / static / index.html ,我们将观察到完全不同的图表:

出色的服务器端API确实非常简洁且易于使用。 但是客户端方面,我们可以使用Java应用程序中的SSE流吗? 答案是肯定的。 JAX-RS 2.1还概述了客户端API,其核心是SseEventSource 。

final WebTarget target = ClientBuilder.newClient().register(JacksonJsonProvider.class).target("http://localhost:8080/services/api/stats/sse");try (final SseEventSource eventSource =SseEventSource.target(target).reconnectingEvery(5, TimeUnit.SECONDS).build()) {eventSource.register(event -> {final Stats stats = event.readData(Stats.class, MediaType.APPLICATION_JSON_TYPE);System.out.println("name: " + event.getName());System.out.println("id: " + event.getId());System.out.println("comment: " + event.getComment());System.out.println("data: " + stats.getLoad() + ", " + stats.getTimestamp());System.out.println("---------------");});eventSource.open();// Just consume SSE events for 10 secondsThread.sleep(10000);
}

如果运行此代码段(假设服务器也已启动并且正在运行),我们将在控制台中看到类似的内容(您可能还记得,数据是随机生成的)。

name: stats
id: 0
comment: null
data: 82, 1509376080027
---------------
name: stats
id: 1
comment: null
data: 68, 1509376081033
---------------
name: stats
id: 2
comment: null
data: 12, 1509376082028
---------------
name: stats
id: 3
comment: null
data: 5, 1509376083028
---------------...

如我们所见,服务器端的OutboundSseEvent变为客户端的InboundSseEvent 。 客户端可以使用通常的MessageBodyReader解析策略,通过指定预期的媒体类型来消耗数据属性中可以反序列化的任何有效负载。

单篇文章中压缩了很多内容。 而且,关于SSE和JAX-RS 2.1的其他事情我们在这里没有涉及,例如使用HttpHeaders.LAST_EVENT_ID_HEADER或配置重新连接延迟。 如果有兴趣学习的话,这些可能是即将发布的帖子的重要话题。

总而言之, JAX-RS对SSE的支持是我们许多人期待已久的事情。 最后,它在那里,请尝试一下!

完整的项目资源可在Github上找到 。

翻译自: https://www.javacodegeeks.com/2017/10/better-late-never-sse-server-sent-events-now-jax-rs.html

jax-rs jax-ws

jax-rs jax-ws_迟来总比没有好:SSE或服务器发送的事件现在已在JAX-RS中相关推荐

  1. 迟来总比没有好:SSE或服务器发送的事件现在已在JAX-RS中

    服务器发送的事件 (或简称为SSE )是非常有用的协议,它允许服务器通过HTTP将数据推送到客户端. 这是我们的网络浏览器支持的年龄,但令人惊讶的是, JAX-RS规范在很长一段时间内都忽略了这一点. ...

  2. 铁总:春运第三日全国铁路发送旅客908.5万人次

    中新网北京1月24日电(杨雨奇)记者从中国铁路总公司获悉,1月23日,在春运开启的第三天,全国铁路发送旅客908.5万人次,同比增加32万人次,增长3.7%.预计24日发送旅客971万人次,同比增长1 ...

  3. 电脑微信发送文件总显示“文件被占用,无法发送,请重新选择。”

    奇怪的问题 在微信发送文件多次提醒时,查看了文件被占用情况,发现并未被占用, 为了解决疑惑,尝试多方发送 同一个文件,发送钉钉就完全没有问题,发送微信却总提示"文件占用,无法发送" ...

  4. 计算机网络数据传输的总时延,【计算机网络】时延、发送时延、传输时延、处理时延、排队时延、时延带宽积...

    时延:指数据从网络的一端传送到另一端所需的时间 发送时延(传输时延):是主机或路由器发送数据帧所需要的时间,也就是从发送数据帧的第一个比特算起,到该帧的最后一个比特发送完毕所需的时间,发送时延 = 数 ...

  5. cxf springmvc_拥抱模块化Java平台:Java 10上的Apache CXF

    cxf springmvc 自Java 9发布最终将Project Jigsaw交付给大众以来,已经过去了整整一年的时间. 这是一段漫长的旅程,但是在那里,所以发生了什么变化? 这是一个很好的问题,答 ...

  6. java cxf_拥抱模块化Java平台:Java 10上的Apache CXF

    java cxf Java 9版本终于将Project Jigsaw交付给大众已经过去了一年的时间. 这是一段漫长的旅程,但是在那里,所以发生了什么变化? 这是一个很好的问题,答案并不明显和直接. 总 ...

  7. 拥抱模块化Java平台:Java 10上的Apache CXF

    Java 9版本终于将Project Jigsaw交付给大众已经过去了一年多的时间. 这是一段漫长的旅程,但是在那里,所以发生了什么变化? 这是一个很好的问题,答案并不明显和直接. 总的来说, 拼图项 ...

  8. DNS安全浅议、域名A记录(ANAME),MX记录,CNAME记录

    相关学习资料 http://baike.baidu.com/link?url=77B3BYIuVsB3MpK1nOQXI-JbS-AP5MvREzSnnedU7F9_G8l_Kvbkt_O2gKqFw ...

  9. GB_T28181-2016.pdf

    国标28181-2016版本检测,由于文件过大,而且博客不支持上传文件,需要GB28181-2016协议文档和公安一所检测文档的可以私信我,QQ:123011785 I C S1 3. 3 1 0 A ...

最新文章

  1. 中国矿业大学计算机英语复试,中国矿业大学2020计算机科学与技术学院考研复试科目参考书目...
  2. 比特币寒冬中,你更应该关注企业区块链!
  3. 8个步骤成功拖垮新业务线!
  4. 一文读懂机器学习中的正则化
  5. 在C#中使用SqlDbType.Xml类型参数
  6. ASP隐藏文件地址,并在下载时替换文件名
  7. JavaScript函数与Window对象
  8. 欧几里得算法扩展欧几里得算法
  9. WPF 3D 基础学习 - 相机、裁剪、光线入门(2)
  10. Android App的架构设计:从VM、MVC、MVP到MVVM
  11. Java Socket实战之四 传输压缩对象
  12. 我的mongoose代码备份
  13. mysql主从复制实践之单数据库多实例
  14. text html mime img,使用Python的email.mime.multipart发送HTML邮件时命...
  15. 数据--第39课 - 二叉树课后练习
  16. 强悍!winrar妙用-将bat脚本打包成exe可执行文件并实现自动执行
  17. w7 internet信息服务器,Win7安装IIS Internet信息服务步骤
  18. python实现快递地址分拣程序(代码有详细注释)
  19. html中icon小图标大全,iconfont字体图标和各种CSS小图标
  20. python数字字母识别_字符图像识别——数字字母混合

热门文章

  1. 模板:2-SAT问题
  2. YbtOJ#20072-[NOIP2020模拟赛B组Day6]相似子串【根号分治】
  3. CF741D-Arpa's letter-marked tree and Mehrdad's Dokhtar-kosh paths【树上启发式合并】
  4. codeforces1454 F. Array Partition
  5. 分布式架构--基本思想汇总
  6. mongodb如何实现更新一个字段的值为另外一个字段的值?
  7. 你必须了解Spring的生态
  8. Maven精选系列--依赖范围、传递、排除
  9. 手把手教你搭建Maven项目
  10. 使用ADO.NET访问数据库