apache cxf_Apache CXF负载平衡和故障转移

apache cxf

不久前,我们已经面临了基于Apache CXF的负载平衡Web服务客户端的需求。 此外,当某些服务器关闭时,客户端应自动进行故障转移。 更糟糕的是,服务器目标地址列表要从外部服务获取并在运行时更新。

最终,我们最终获得了本地开发的负载平衡微库(ESB / UDDI / WS-Addressing似乎是一个有趣的选择,但在我们的情况下这是一个过大的选择)。 是否仅知道Apache CXF已经(几乎)已经支持所有这些功能?

不过,不要怪我们,仅提及此功能会指向非常糟糕的文档页面(如果您将404称为“差”)。 如果它不在正式文档中,我希望可以在Apache CXF Web服务开发书中找到它-不幸的是,那里也很不幸。 但是,嘿,您自己探索这些功能不是更有趣吗? 这是我们开始的客户端配置:
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:jaxws="http://cxf.apache.org/jaxws"xmlns:clustering="http://cxf.apache.org/clustering"xmlns:util="http://www.springframework.org/schema/util"><jaxws:client id="testServiceClient"serviceClass="com.blogspot.nurkiewicz.cxfcluster.SimpleService"address="http://serverA/simple"></jaxws:client></beans>
端点接口在这里并不重要,足以知道将testServiceClient注入了其他服务,并且负载平衡和故障转移功能不应影响现有代码。 请注意,服务地址是固定的且经过硬编码的(当然可以将其外部化并在启动时读取)。
令人惊讶的是,仅启用故障转移是非常简单,直接和不言自明的(尽管是XML):
<jaxws:client id="testServiceClient"serviceClass="com.blogspot.nurkiewicz.cxfcluster.SimpleService"address="http://serverA/simple"><jaxws:features><clustering:failover><clustering:strategy><bean class="org.apache.cxf.clustering.RandomStrategy"><property name="alternateAddresses"><util:list><value>http://serverB/simple</value><value>http://serverC/simple</value><value>http://serverD/simple</value></util:list></property></bean></clustering:strategy></clustering:failover></jaxws:features></jaxws:client>
serverA地址用作主要端点,但是当它失败时, 将以随机顺序检查所有故障转移端点( serverB , serverC和serverD )。 为了发挥这种配置的作用,我建议您打开Apache CXF请求和响应日志记录 :
<cxf:bus><cxf:features><cxf:logging/></cxf:features>
</cxf:bus>
再次(!),官方文档中没有提到非常方便的配置参数prettyLogging ,该参数可以应用于日志记录功能,以便在记录之前正确格式化XML请求和响应( 换行和缩进)。 我不建议在生产设置中使用它,但是在开发和测试过程中,格式化SOAP消息是非常宝贵的:
<bean id="abstractLoggingInterceptor" abstract="true"><property name="prettyLogging" value="true"/>
</bean>
<bean id="loggingInInterceptor" class="org.apache.cxf.interceptor.LoggingInInterceptor" parent="abstractLoggingInterceptor"/>
<bean id="loggingOutInterceptor" class="org.apache.cxf.interceptor.LoggingOutInterceptor" parent="abstractLoggingInterceptor"/><cxf:bus><cxf:inInterceptors><ref bean="loggingInInterceptor"/></cxf:inInterceptors><cxf:outInterceptors><ref bean="loggingOutInterceptor"/></cxf:outInterceptors><cxf:outFaultInterceptors><ref bean="loggingOutInterceptor"/></cxf:outFaultInterceptors><cxf:inFaultInterceptors><ref bean="loggingInInterceptor"/></cxf:inFaultInterceptors>
</cxf:bus>
因此,如果主要端点不可用,我们的服务很好地故障转移到备份端点。 但是我们有四个等效的服务器,我们希望我们的客户以相同的概率(轮循,随机?)平等地对待它们。 这是负载均衡进入阶段的时候:
<jaxws:client id="testServiceClient" serviceClass="com.blogspot.nurkiewicz.cxfcluster.SimpleService"><jaxws:features><clustering:loadDistributor><clustering:strategy><bean class="org.apache.cxf.clustering.SequentialStrategy"><property name="alternateAddresses"><util:list><value>http://serverA/simple</value><value>http://serverB/simple</value><value>http://serverC/simple</value><value>http://serverD/simple</value></util:list></property></bean></clustering:strategy></clustering:loadDistributor></jaxws:features></jaxws:client>
请注意,客户端本身不再定义地址属性。 这表明, alternateAddresses列表仅在所有调用中使用,并且不存在主地址-实际上是这种情况。 SequentialStrategy将使用一个端点接另一个端点,以提供良好的循环实现(也提供RandomStrategy )。 同样,在此配置中,您将免费获得故障转移–如果任何端点发生故障,将从第一个端点开始的所有端点都将受到检查(显然,刚刚发生故障的端点除外)。
大! 现在,CXF客户端变得更加僵化和容错。 但是,在我们追求更高可用性并最大程度地减少停机时间的过程中,仅在应用程序启动时才加载备用节点(换句话说,添加新服务器需要所有客户端重新启动)实在太过局限了。 幸运的是,我们可以通过两个简单的步骤使负载均衡更加动态。
<jaxws:client id="testServiceClient" serviceClass="com.blogspot.nurkiewicz.cxfcluster.SimpleService"><jaxws:features><clustering:loadDistributor><clustering:strategy><bean class="org.apache.cxf.clustering.SequentialStrategy"><property name="alternateAddresses" ref="alternateAddresses"/></bean></clustering:strategy></clustering:loadDistributor></jaxws:features></jaxws:client><util:list id="alternateAddresses" list-class="java.util.concurrent.CopyOnWriteArrayList"><value>http://serverA/simple</value><value>http://serverB/simple</value><value>http://serverC/simple</value><value>http://serverD/simple</value>
</util:list>
没什么,提取嵌套的匿名bean。 但是可以访问此列表(请注意,我使用了java.util.concurrent.CopyOnWriteArrayList )使我们可以将其注入任何其他服务,从而可能改变其状态。 我怎么知道这行得通? 好吧,我花了几个下午来调试Apache CXF,以最终发现负载平衡算法:第一次调用时,CXF要求策略提供可能节点的列表。 然后它会将这个列表返回策略要求到(在这里小编WTF ......)挑选一个战略决定哪些地址,使用和删除列表中挑选地址(另一个小一个在这里...)当CXF发现列表是空的,故事重复本身。 因此,如果我们在运行时替换备用地址列表,则一轮新列表将返回到核心CXF基础结构。
因为我是JMX的拥护者,所以这是我们修改地址列表的方法(您可以使用任何喜欢的机制):
@Service
@ManagedResource
public class AlternateAddressesManager {@Resourceprivate List alternateAddresses;@ManagedOperationpublic void addAlternateAddress(String address) {alternateAddresses.add(address);}@ManagedOperationpublic boolean removeAlternateAddress(String address) {return alternateAddresses.remove(address);}@ManagedAttributepublic List getAlternateAddresses() {return Collections.unmodifiableList(alternateAddresses);}}
是的,这是SequentialStrategy使用的非常相同的alternateAddresses列表,因此,只需对其进行修改,就可以更改CXF寻址行为。 可以说,我们可以扩展CopyOnWriteArrayList,添加一些额外的启用JMX的方法(或者,探究Springs的灵活性,直接通过JMX公开List方法!),但这会降低可维护性,我认为这是较差的设计。
最后,我们可以按照下面的屏幕快照启动jconsole或JVisualVM,并享受我们的负载平衡基础架构:
快乐? 并不是的。 在研究CXF源代码时,我遇到了关于LoadDistributorFeature和FailoverTargetSelector类的可怕的JavaDoc注释,它们在负载平衡过程中起着重要的作用:
/ **
* […]
*请注意,此功能会即时更改导管,从而使
* 客户端不是线程安全的。
* /
重点放在粗体的文本上(好的,老实说,我不理解其余内容)。 如果您已经使用Spring一段时间,您就会知道testServiceClient bean是多个线程同时使用的共享单例(否,使其原型作用域无济于事;为什么?),这与默认的EJB无状态会话bean相反,被汇集。 幸运的是,Spring也为此提供了一个内置的解决方案。 但是在我最终提出正确的解决方案之前,出现了一些障碍。
首先,来自CXF名称空间的jaxws:client标记不允许定义bean范围,默认情况下为单例,而我们要合并客户端。 因此,我不得不使用org.apache.cxf.jaxws.JaxWsProxyFactoryBean恢复到良好的旧bean定义。 没问题,稍微冗长些,尽管如果您不希望使用自定义Spring名称空间,则可能一开始就使用它。 现在最好的部分是:我可以将具有原型范围的任何bean封装在特殊的代理中,Spring会自动创建一个对象池(基于commons-pool库),并根据需要创建尽可能多的bean实例,以使每个bean仅由一个使用。线。 配置如下:
<bean id="testServiceClientFactoryBean" class="org.apache.cxf.jaxws.JaxWsProxyFactoryBean"><property name="serviceClass" value="com.blogspot.nurkiewicz.cxfcluster.SimpleService"/><property name="features"><util:list><bean class="org.apache.cxf.clustering.LoadDistributorFeature"><property name="strategy"><bean class="org.apache.cxf.clustering.SequentialStrategy"><property name="alternateAddresses" ref="alternateAddresses"/></bean></property></bean></util:list></property>
</bean><bean id="testServiceClientTarget" factory-bean="testServiceClientFactoryBean" factory-method="create" scope="prototype" lazy-init="true"/><bean id="testServiceClient" class="org.springframework.aop.framework.ProxyFactoryBean"><property name="targetSource"><bean class="org.springframework.aop.target.CommonsPoolTargetSource"><property name="targetClass" value="com.blogspot.nurkiewicz.cxfcluster.SimpleService"/><property name="targetBeanName" value="testServiceClientTarget"/><property name="maxSize" value="10"/><property name="maxWait" value="5000"/></bean></property>
</bean>
您是否注意到maxSize和maxWait池属性? 他们真是太酷了 ! 您可以告诉Spring在池中创建的客户端不要超过10个,并且如果池为空(当前正在使用所有Bean),我们应该等待不超过5000毫秒(之后发生的事情是可配置的!)非常简单但功能强大的限制机制,比JMS或显式线程池简单得多,我们绝对免费! 例如,不想服务超过20个并发Web服务客户端? 使服务器端点访问服务Bean的大小限制为20。超过此限制的客户端将被拒绝,因为没有可用的服务Bean。
当然,在成人世界中,没有任何效果可以预期。 我很快发现JaxWsProxyFactoryBean.create不是线程安全的, 并报告了CXF-3558 。 作为解决方法,我必须通过将其 子类化 来同步 CommonsPoolTargetSource 使用的客户端工厂 :
import org.apache.commons.pool.ObjectPool;
import org.apache.commons.pool.PoolUtils;
import org.springframework.aop.target.CommonsPoolTargetSource;public class SynchCommonsPoolTargetSource extends CommonsPoolTargetSource {@Overrideprotected ObjectPool createObjectPool() {return PoolUtils.synchronizedPool(super.createObjectPool());}}
似乎需要同步工厂,所以我创建了SPR-8382-也许它将找到正式发行的方式。 顺便说一句,在撰写本文时,我还报告了IDEA-70365 – 报告了List类型bean的 虚假 “无法自动装配”错误
最后! 我们的负载平衡和故障转移就像一个魅力。 下一步将是暂时丢弃掉了几秒钟的节点,如果此后端点仍然掉线,则增加此时间。 但是Apache CXF在这一领域的API如此糟糕,以至于我不得不离开这个话题一段时间。 也许您可以帮忙?

参考: NoBlogDefFound博客上的 JCG合作伙伴 Tomasz 在Apache CXF中启用负载平衡和故障转移 。

相关文章 :
  • 具有Spring和Maven教程的JAX–WS
  • Spring 3 RESTful Web服务
  • Tomcat 7上具有RESTeasy JAX-RS的RESTful Web服务– Eclipse和Maven项目

翻译自: https://www.javacodegeeks.com/2011/06/apache-cxf-load-balancing-failover.html

apache cxf

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/359675.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

HDU 1874 最直接的最短路径问题

题目链接&#xff1a;http://acm.hdu.edu.cn/showproblem.php?pid1874 Problem Description某省自从实行了很多年的畅通工程计划后&#xff0c;终于修建了很多路。不过路多了也不好&#xff0c;每次要从一个城镇到另一个城镇时&#xff0c;都有许多种道路方案可以选择&#xf…

领域驱动设计模式设计与实践_在域驱动设计中使用状态模式

领域驱动设计模式设计与实践域驱动设计&#xff08;DDD&#xff09;是一种开发软件的方法&#xff0c;其中&#xff0c;通过将实现与核心业务概念的不断发展的模型相联系&#xff0c;解决了问题的复杂性。 该术语是由Eric Evans创造的&#xff0c;并且有一个DDD专用站点可以促进…

html 英文文字纵向排列,CSS几种简单方法实现文字竖向排版

1.一个句子的竖向排列如图&#xff1a;1.2. test.one {width: 20px;margin: 0 auto;line-height: 24px;font-size: 20px;}.two {width: 15px;margin: 0 auto;line-height: 24px;font-size: 20px;word-wrap: break-word;/*英文的时候需要加上这句&#xff0c;自动换行*/}我是竖列…

jstree 节点拖拽保存数据库

需要jstree具有拖拽功能需要在加载jstree时添加dnd插件&#xff0c;具体看代码&#xff1a; $(**).jstree({//plugins-各种jstree的插件引入&#xff0c;展示树的多样性 plugins : [ "dnd", "types", "wholerow" ], core : {"check_callbac…

自动添加html结束标志,HTML:包含或排除可选的结束标记?

MYYA我在这里添加一些链接来帮助您了解HTML的历史&#xff0c;以便您了解各种矛盾。这不是你的问题的答案&#xff0c;但在阅读这些各种摘要后你会知道更多。我们是怎么来到这里的&#xff1f; - 潜入HTML5网络历史HTML简史HTML的历史 - HTML WG WikiDive Into HTML5的一些摘录…

JAR清单类路径不仅适用于Java Application Launcher

自从我开始学习Java以来​​&#xff0c;我几乎已经知道&#xff0c; 清单文件中的Class-Path标头字段为可执行JAR &#xff08;具有由另一个称为Main-Class清单指定应用程序起点的 JAR&#xff09;指定相对运行时类路径。 一个同事最近碰到一个让我感到惊讶&#xff0c;因为它…

通过url,获取html内容,并解析,如何使用 JavaScript 解析 URL

在 Web 开发中&#xff0c;有许多情况需要解析 URL&#xff0c;这篇主要学习如何使用 URL 对象实现这一点。开始创建一个以下内容的 HTML 文件&#xff0c;并在浏览器中打开。JavaScript URL parsing// 激动人心的代码即将写在这里如果你想尝试本文中的任何内容&#xff0c;可以…

正确使用计算机说课稿,《计算机结构原理初步》说课稿

在教师招聘考试的过程中&#xff0c;高中信息说课稿的难度就在于如何处理理论与实践的关系&#xff0c;希望这篇《计算机结构原理初步》说课稿能给予你帮助。各位考官大家好!我是号考生&#xff0c;今天我说课的题目是《计算机结构原理初步》。现代教学理论认为&#xff0c;在教…

计算机2013知识,2013年全国计算机一级考试B基本知识点五

基础5单元格操作对已建立的工作表&#xff0c;根据需要可以编辑修改其中的数据首先要移动单元格指针到目的地或选定编辑对象&#xff0c;然后才能进行增、删、改操作。1.单元格指针的移动要编辑某单元格&#xff0c;必须把单元格指针移动到该单元格&#xff0c;使之成为当前单元…

smartgwt_高级SmartGWT教程,第1部分

smartgwt贾斯汀&#xff08;Justin&#xff09;&#xff0c;帕特&#xff08;Pat&#xff09;和我已经开始着手进行一个需要管理和管理用户界面的副项目。 在与SmartGWT和GWT共同工作了一段时间之后&#xff0c;我们决定使用SmartGWT创建接口。 我们非常喜欢视觉组件&#xff0…

计算机英语女人英语怎么说,英语时差:计算机和女人

00:0000:00微信扫码登陆&#xff0c;畅听全站所有音频&#xff01;(20秒后自动关闭)X关注后&#xff0c;点此关闭https://online2.tingclass.net/lesson/shi0529/10000/10183/67.mp3https://image.tingclass.net/statics/js/2012When you hear the term, "computer geek,&…

云计算系统是大规模计算机系统吗,云计算的系统架构及技术探析

云计算技术属于计算机技术的一种&#xff0c;是目前计算机技术中应用以及研究重点之一&#xff0c;那么云计算到底是什么呢&#xff1f;云计算是在并行处理&#xff0c;分析式处理等技术的基础上发展而来的新技术&#xff0c;可以有效的将计算机进行整合&#xff0c;建立新颖的…

Vue 封装echarts柱状图(Bar)组件

目的&#xff1a;减少重复代码&#xff0c;便于维护 显示效果 组件代码 <template><div class"ldw-data-content-box"><div class"ldw-chilren-box"><div class"title" v-if"title">{{ title }}</div>…

smartgwt_高级SmartGWT教程,第2部分

smartgwt这是我的教程的第二部分&#xff0c;有关使用SmartGWT快速进行UI开发。 在本教程的第一部分中 &#xff0c;我们创建了基本的界面布局并添加了一些基本组件。 现在是时候解决这个问题&#xff0c;并使用SmartGWT的真正功能了。 在继续之前&#xff0c;让我们记住到目前…

百年理工计算机专业课程,这两所国内的百年理工院校,实力强劲,都是国内顶尖实力...

2021年全国高考终于圆满落下了帷幕&#xff0c;但考生们即将要面临一件人生大事&#xff0c;那就是高考志愿填报&#xff0c;因此高考虽然结束&#xff0c;但考生们的压力仍然是不小&#xff0c;只有正确地填好高考志愿&#xff0c;确保自己能够被心仪的高校收录取&#xff0c;…

数学图形之SineSurface与粽子曲面

SineSurface直译为正弦曲面.这有可能和你想象的正弦曲线不一样.如果把正弦曲线绕Y轴旋转,得到的该是正弦波曲面.这个曲面与上一节中的罗马曲面有些相似,那个是被捏过的正四面体,这个则是个被捏过正方体. 本文将展示SineSurface与粽子曲面的生成算法和切图,使用自己定义语法的脚…

计算机工程学院文艺例会,西航职院 | 计算机工程学院 2019年度学生会干部第一次例会...

原标题&#xff1a;西航职院 | 计算机工程学院 2019年度学生会干部第一次例会西航职院计算机工程学院&#xff0c;2019年度第一次学生会干部例会&#xff0c;于2月26日下午在四号教学楼201顺利召开。参加此次会议的有计算机学生会全体学生干部和指导老师刘老师。会议开始&#…

dateformat线程_Java最佳实践–多线程环境中的DateFormat

dateformat线程这是有关使用Java编程语言时的拟议实践的系列文章的第一篇。 所有讨论的主题均基于用例&#xff0c;这些用例源于电信行业关键任务超高性能生产系统的开发。 在阅读本文的每个部分之前&#xff0c;强烈建议您参考相关的Java API文档以获取详细信息和代码示例。…

关于图连通性的几道题(水)

POJ 2186 强连通分量缩点 1 #include<cstdio>2 #include<cstring>3 #include<algorithm>4 using namespace std;5 6 int en[10010], col[10010], dfn[10010], low[10010], stack[10010], tot[10010], chu[10010];7 bool ins[10010];8 int n, m, esize, dtime…

一台计算机硬盘容量标为800gb,一台计算机的硬盘容量标为800GB,其存储容量是()。...

台硬盘不适的装用于料是室外饰材。行的有_能进操作&#xff0c;计算机系统源管在W“资”中操作理器。不正有_确的&#xff0c;容量下列叙述中。的有描述正确&#xff0c;其存文输的中s系统下关于入。文档选定应_整篇&#xff0c;储容d文档编在W辑中。不能作进行_操&#xff0c;…