Java EE应用程序的单片到微服务重构

您是否曾经想过将现有的Java EE整体应用程序重构为基于微服务的应用程序需要做什么?

该博客解释了一个简单的购物车示例如何转换为基于微服务的应用程序,以及围绕它的一些担忧。 整体和基于微服务的应用程序的完整代码库位于: github.com/arun-gupta/microservices 。

继续阅读以获取更多荣耀!

Java EE Monolith

Java EE整体应用程序通常定义为WAR或EAR档案。 该应用程序的全部功能都打包在一个单元中。 例如,在线购物车可能包含用户,目录和订单功能。 所有网页都在应用程序的根目录中,所有相应的Java类都在WEB-INF/classes目录中,资源在WEB-INF/classes/META-INF目录中。

让我们假设您的整体设计不是设计为一个分散的大泥巴,并且该应用程序是按照良好的软件体系结构构建的。 一些常见的规则是:

  • 分离关注点,可能使用Model-View-Controller
  • 使用定义明确的API的高内聚和低耦合
  • 不要重复自己(干)
  • 接口/ API和实现是分开的,并且遵循Demeter定律 。 类不会直接调用其他类,因为它们恰好位于同一存档中
  • 使用域驱动设计将与域/组件相关的对象保持在一起
  • YAGNI或您不需要它:不要构建您现在不需要的东西

这是一个琐碎的购物车整体WAR存档的外观:

javaee-monolithic

此整体应用程序具有:

  • 网页,例如.xhtml文件,用于用户,目录和订单组件,打包在归档的根目录中。 这些网页还打包了在不同网页之间共享的所有CSS和JavaScript资源。
  • 这三个组件的类在WEB-INF/classes目录中的单独软件包中。 多个类使用的所有实用程序/公共类也都打包在这里。
  • 每个组件的配置文件打包在WEB-INF/classes/META-INF目录中。 该应用程序的任何配置文件(例如分别连接和填充数据存储的persistence.xmlload.sql也都打包在此处。

它具有众所周知的体系结构,IDE友好,易于共享,简化测试,易于部署等常见优点 。 但同时也具有诸如敏捷性受限,持续交付的障碍,技术堆栈“卡住”,技术债务增加等缺点 。

即使微服务如今已成为热门,但整体还是不错的。 即使那些对您不起作用的服务,也可能不会立即迁移到微服务中,从而受益匪浅。 其他方法,例如更好的软件工程和体系结构,可能会有所帮助。 微服务既不是免费的午餐也不是灵丹妙药,它需要大量的投资才能成功,例如服务发现,服务复制,服务监视,容器,PaaS,弹性等等。

除非您的系统过于复杂而无法作为一个整体进行管理,否则甚至不要考虑使用微服务。

微服务高级版

Java EE的微服务架构

好了,我已经听说了所有这些内容,但是想知道之前/之后的内容,即整体代码库的外观以及重构后的微服务代码库的外观。

首先,让我们看一下整体架构:

javaee-microservices

该体系结构的关键部分是:

  • 在将用户,订单和目录组件打包为单独的WAR文件的情况下,应在功能上分解应用程序。 每个WAR文件应具有该组件所需的相关网页( #15 ),类和配置文件。
    • Java EE用于实现每个组件,但是对堆栈没有长期承诺,因为不同的组件使用定义良好的API( #14 )相互通信。
  • 每个档案都有自己的数据库,即不共享数据存储。 这允许每个微服务发展和选择最合适的数据存储类型-关系,NoSQL,平面文件,内存中或其他。
  • 每个组件都将向服务注册中心注册。 这是必需的,因为每个服务的多个无状态实例可能在给定的时间运行,并且它们的确切端点位置仅在运行时才知道( #17 )。 Netflix Eureka , Etcd , Zookeeper是该领域的一些选项( 更多详细信息 )。
  • 如果组件之间需要相互通信(这很常见),则可以使用预定义的API进行通信。 实现同步的REST或实现异步通信的Pub / Sub是实现此目的的常用方法。在我们的案例中,Order组件发现用户和目录服务,并使用REST API与它们进行对话。
  • 该应用程序的客户端交互是在另一个应用程序(在本例中为Shopping Cart UI)中定义的。 该应用程序主要从服务注册表中发现服务并将其组合在一起。 它应该主要是一个哑代理,在其中调用不同组件的UI页面以显示界面( #18 )。可以通过提供标准CSS / JavaScript资源来实现常见的外观。

该应用程序相当琐碎,但至少强调了一些基本的体系结构差异。

整体与微服务

下面比较了基于单体和基于微服务的应用程序的一些统计信息:

特性 整体式 微服务
档案数量 1个 5
  • 合同(JAR,〜4 KB)
  • 顺序(WAR,〜7 KB)
  • 用户(WAR,〜6 KB)
  • 目录(WAR,〜8 KB)
  • Web UI(WAR,27 KB)
网页 8 8(见下文)
配置文件 4
  • web.xml
  • template.xhtml
  • persistence.xml
  • load.sql
每个档案3个
  • persistence.xml
  • load.sql
  • web.xml
类文件 12 26
  • 每个档案的服务注册
  • 服务发现类
  • 每个档案的应用程式类别
档案总大小 24 KB 〜52 KB(总计)

  • 整体应用程序的代码库位于: github.com/arun-gupta/microservices/tree/master/monolith/everest
  • 支持微服务的应用程序的代码库位于: github.com/arun-gupta/microservices/tree/master/microservice

问题和待办事项

以下是将整体重构为基于微服务的应用程序时遇到的问题和TODO:

  • Java EE已经允许使用EAR打包对应用程序进行功能分解。 应用程序的每个组件都可以打包为WAR文件,并捆绑在EAR文件中。 他们甚至可以通过这种方式共享资源。 现在这不是真正的微服务方式,但这可能是使您入门的过渡步骤。 但是,请注意, @FlowScoped bean没有在EAR中正确激活( WFLY-4565 )。
  • 使用JSF资源库模板提取所有模板文件。
    • 目前,所有网页都在everest模块中,但它们应该位于每个组件中( #15 )。
  • 将整体数据库拆分为多个数据库需要为每个应用程序使用单独的persistence.xml和DDL / DML脚本。 同样,需要相应地创建迁移脚本,例如使用Flyway。
  • 必须为所有组件创建一个REST接口,该接口需要被另一个组件访问。
  • UI仍在单个Web应用程序中。 而是应将其包含在已分解的WAR( #15 )中,然后再在哑代理中进行组合。 闻起来像portlet吗?
  • 在PaaS中部署多个WAR文件( #12 )
  • 每个微服务都应易于部署在容器中( #6 )

这是单片应用程序的类的完整列表:

./target/classes/org/javaee7/wildfly/samples/everest/cart/Cart.class
./target/classes/org/javaee7/wildfly/samples/everest/cart/CartItem.class
./target/classes/org/javaee7/wildfly/samples/everest/catalog/CatalogItem.class
./target/classes/org/javaee7/wildfly/samples/everest/catalog/CatalogItemBean.class
./target/classes/org/javaee7/wildfly/samples/everest/catalog/CatalogItemType.class
./target/classes/org/javaee7/wildfly/samples/everest/checkout/Order.class
./target/classes/org/javaee7/wildfly/samples/everest/checkout/OrderBean.class
./target/classes/org/javaee7/wildfly/samples/everest/checkout/OrderItem.class
./target/classes/org/javaee7/wildfly/samples/everest/checkout/Shipping.class
./target/classes/org/javaee7/wildfly/samples/everest/uzer/Uzer.class
./target/classes/org/javaee7/wildfly/samples/everest/uzer/UzerBean.class
./target/classes/org/javaee7/wildfly/samples/everest/uzer/UzerItem.class

这是基于微服务的应用程序的类的完整列表:

./catalog/target/classes/org/javaee7/wildfly/samples/everest/catalog/ApplicationConfig.class
./catalog/target/classes/org/javaee7/wildfly/samples/everest/catalog/CatalogItem.class
./catalog/target/classes/org/javaee7/wildfly/samples/everest/catalog/CatalogItemREST.class
./catalog/target/classes/org/javaee7/wildfly/samples/everest/catalog/CatalogItemType.class
./catalog/target/classes/org/javaee7/wildfly/samples/everest/catalog/ServiceRegistration.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/cart/Cart.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/cart/CartItem.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/catalog/CatalogBean.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/catalog/CatalogItem.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/checkout/Order.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/checkout/OrderBean.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/checkout/OrderItem.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/checkout/Shipping.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/ServiceDiscovery.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/ServiceDiscoveryStatic.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/ServiceDiscoveryURI.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/ServiceDiscoveryZooKeeper.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/uzer/UzerBean.class
./everest/target/classes/org/javaee7/wildfly/samples/everest/uzer/UzerItem.class
./order/target/classes/org/javaee7/wildfly/samples/everest/order/ApplicationConfig.class
./order/target/classes/org/javaee7/wildfly/samples/everest/order/Order.class
./order/target/classes/org/javaee7/wildfly/samples/everest/order/OrderItem.class
./order/target/classes/org/javaee7/wildfly/samples/everest/order/OrderREST.class
./user/target/classes/org/javaee7/wildfly/samples/everest/uzer/ApplicationConfig.class
./user/target/classes/org/javaee7/wildfly/samples/everest/uzer/UserREST.class
./user/target/classes/org/javaee7/wildfly/samples/everest/uzer/Uzer.class
  • 同样,完整的代码库位于github.com/arun-gupta/microservices 。

未来话题

本系列中的一些未来主题将是:

  • 微服务是否需要容器?
  • 如何使用容器部署多个微服务?
  • 如何轻松监控所有这些服务?
  • A / B测试
  • 使用微服务和容器进行连续部署

您还想看些什么?

请享用!

翻译自: https://www.javacodegeeks.com/2015/06/monolithic-to-microservices-refactoring-for-java-ee-applications.html

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

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

相关文章

Java代码韦伯分布_第十五节、韦伯局部描述符(WLD,附源码)

纹理作为一种重要的视觉线索,是图像中普遍存在而又难以描述的特征,图像的纹理特征一般是指图像上地物重复排列造成的灰度值有规则的分布。纹理特征的关键在于纹理特征的提取方法。目前,用于纹理特征提取的方法有很多,最具有代表性…

JavaFX真实世界应用程序:欧洲电视网广播联盟

我荣幸地与今年在德累斯顿Saxonia 系统公司工作的Alexander Casall一起在JavaOne上展示了多个JavaFX Real-World应用程序。 在准备本次演讲时,我们向客户和合作伙伴发送了一份调查表,并向他们询问了一些与他们的项目,他们的应用程序以及他们对…

java 三大集合_java中的三大集合入门笔记(简单实用)

为什么使用集合框架如图:集合的好处:集合弥补了数组的缺陷,它比数组更灵活更实用,可大大提高软件的开发效率,而且不同的集合适用于不同的场合。Java集合主要分为以下3种类型Java集合框架提供了一套性能优良、使用方便的…

Java面试题2019简书_2019最新Spring面试题大全含答案之Spring Beans(2019最全Spring超级葵花宝典)...

1.什么是Spring beans?Spring beans 是那些形成Spring应用的主干的java对象。它们被Spring IOC容器初始化,装配,和管理。这些beans通过容器中配置的元数据创建。比如,以XML文件中 的形式定义。Spring 框架定义的beans都是单件bean…

使用此首选项可加快Eclipse m2e配置

谁不认识他们。 Eclipse中的旧式JFace对话框可以使您直观地看到实际上是一个相当简单的XML或属性文件。 对于m2e,它看起来像这样: 不幸的是,此屏幕的加载速度有点慢,除了检查版本号和您将永远不会更改的其他内容之外&#xff0c…

【百度地图API】发布静态图API啦!只需一个网址,即可展示定制百度地图!

【百度地图API】发布静态图API啦!只需一个网址,即可展示定制百度地图! 原文:【百度地图API】发布静态图API啦!只需一个网址,即可展示定制百度地图!摘要: 百度地图静态图API!您无须执…

脚本在流程中的性能影响

我们经常看到人们出于各种目的而使用脚本(例如,在服务任务,执行侦听器等中)。 使用脚本和Java逻辑通常很有意义: 它不需要打包到jar中并放在classpath上 它使流程定义更易于理解:无需查看其他文件 逻辑是…

PowerDesigner使用教程

原文:http://www.cnblogs.com/huangcong/archive/2010/06/14/1757957.html PowerDesigner是一款功能非常强大的建模工具软件,足以与Rose比肩,同样是当今最著名的建模软件之一。Rose是专攻UML对象模型的建模工具,之后才向数据库建模…

红帽峰会2015所需的JBoss BPM内容指南

明年再见? 今年在Red Hat Summit上,我们在JBoss BRMS和JBoss BPM Suite演讲中获得了很多乐趣。 在DevNation周围也有一些社区会议,重点介绍了使我们的产品成为可能的项目。 您可以在他们的博客上找到此演讲的概述,并在其中提供…

设计模式 之 享元

享元模式(Flyweight) 运用共享技术有效地支持大量细粒度的对象。 还记得那年夏天一起在作文本上玩过的五子棋吗?五子棋是一种两人对弈的纯策略型棋类游戏,它起源于中国古代的传统黑白棋种之中的一个,不…

php制作简单的用户登陆,如何用php代码实现简单的用户登陆以及登陆验证功能

本文主要简单讲述了如何使用php实现简单的用户登陆以及登陆验证效果。1、首先实现通过Session实现用户的登录那么什么是session呢?session被译为会话,它主要是为了在一定访问期间在不同页面之间间传输数据,用来解决http协议无状态。session在…

Quick cocos2dx-Lua(V3.3R1)学习笔记(十)-----搭建安卓打包环境,用官方示例anysdk生成apk运行...

话说我这一篇就写搭建打包安卓环境,是不是有点过早了(其实我主要是怕以后重装系统,忘了怎么搭建了),但是迟早要面对的痛苦,一直延后也不是办法,对吧。 在官方文档中,对于打包安卓apk…

php实现返回顶部,返回顶部js

返回顶部jsfunction goTopEx(){var objdocument.getElementById(“goTopBtn”);function getScrollTop(){return (document.documentElement.scrollTopdocument.body.scrollTop);}function setScrollTop(value){if(document.documentElement && document.documentEleme…

中国大学MOOC-数据结构基础习题集、06-2、旅游规划

题目链接:http://www.patest.cn/contests/mooc-ds/06-2 题目分析:陈姥姥说,这是Dijstra算法的一道题。题目是中文的,这里就不再啰嗦了。有一点提示一下,咱们平时用的Dijistra算法,是用来求最短路径的。这道…

iOS安全攻防(三):使用Reveal分析他人app

使用Reveal分析他人app准备工作1)已越狱的设备,而且已安装了OpenSSH,MobileSubstrate等有用工具(Cydia源里安装)2)本地已安装了Reveal操作步骤1)拷贝framework和dylib到越狱机scp -r /Applications/Reveal.app/Contents/SharedSup…

JBoss BPM Travel Agency演示与现代BPM数据集成

不久前,我们启动了一个规模较大的JBoss Travel Agency演示项目,以展示JBoss BPM Suite的一些更有趣的功能。 我们提供了一系列视频 ,不仅向您展示了如何安装它,项目中各种规则和流程工件的含义,还向您介绍了在实际使用…

scala入门-10 隐式转换、隐式参数、隐式类

到目前为止,隐式转换是scala的重点和难点了,加油~ 我们先创建一个类名称叫Implicit.scala 再看一个隐式参数的例子: 上面的例子中使用了隐式参数,我们也可以明显的指明参数: 下面看一下隐式类: 相当于&…

连载《一个程序猿的生命周期》-2.城市校园生活

一个程序猿的生命周期 微信平台 口 号:职业交流,职业规划;面对现实,用心去交流、感悟。 公众号:iterlifetime 百木-ITer职业交流奋斗 群:141588103 微 博:http://www.weibo.com/u/57234…

excel使用MySQL数据,如何使用mysql完成excel中的数据生成

Excel是数据分析中最常用的工具,本篇文章通过mysql与excel的功能对比介绍如何使用mysql完成excel中的数据生成,数据清洗,预处理,以及最常见的数据分类,数据筛选,分类汇总,以及数据透视等操作。本…

Spring Batch –使用JavaConfig替换XML作业配置

我最近协助一个客户启动并运行了Spring Batch实现。 该团队决定继续为批处理作业使用基于JavaConfig的配置,而不是传统的基于XML的配置。 随着这越来越成为配置Java应用程序的一种常用方法,我觉得是时候更新Keyhole的Spring Batch系列了 ,向您…