Java理论与实践: 让J2EE脱离容器

在大多数情况下,Java 应用程序要么是 J2EE 应用程序、要么是 J2SE 应用 程序,并且在这一点上是泾渭分明的。J2EE 应用程序需要 J2EE 容器的服务, 容器要实现一长串的 J2EE API,包括 Enterprise JavaBean (EJB)、JTA、JNDI 、JMS、JCA 和 JMX。J2EE API 设计为协同工作;毕竟,J2EE 设计是从多年来 数百人开发企业应用程序的经验中提取出的公共需求。像所有框架一样,J2EE API 的主要目的是“不重新发明轮子”。

有一些 API 属于 J2EE 规范的一部分,但是可以很容易地在 J2SE 应用程序 中使用,如 JDBC、JSP 和 servlet,但是对于大多数 J2EE API,J2EE 是一个 要么是要么不是的命题--大多数 J2EE API 需要全功能的 J2EE 容器。不过,有 一些服务器应用程序开发为 J2SE 应用程序而非 J2EE 应用程序,这通常都有很 好的理由。为什么这些应用程序的开发人员必须重新发明轮子呢?J2EE 中是否 有部分内容可以容易地被 J2SE 应用程序借用来提供同样的优点呢?什么组件可 以同时用于 J2EE 和 J2SE 应用程序的组件呢?

松散耦合

J2EE 的一个主要设计原理是 J2EE 应用程序可以松散地耦合--用组件组装, 在组装或者部署应用程序时而不是在组件开发时定义或者改变这些组件的相互连 接。J2EE 组件使用 JNDI 相互查找和查找所需要的资源,如 JDBC 和 JMS 连接 。JMS 这样的技术鼓励松散耦合,允许灵活地为工作流程建模、容易分配处理任 务、可伸缩性和容错性。很多 J2SE 服务器应用程序也可以从这些技术和原理中 受益。

像 JDBC、JMS 和 JNDI 这样的 API 基本上是“中间件”--它们作为应用程 序与不同的服务提供者之间的统一接口。几乎每一个数据库服务器都有 JDBC 驱 动程序,有大量的免费数据库服务器,所以几乎每一个希望利用数据库的 Java 应用程序都可以容易地做到这一点。不过,对于 JMS 就不是这样了。消息队列 服务器远没有数据库这样常见,特别是在小型商店中。但是有大量的应用程序可 以通过使用消息队列而极大地受益。

Somnifugi JMS

消息队列是一个功能强大的范例,它用于构建健壮的、灵活的、松散耦合的 、可伸缩的应用程序。有一些商业消息队列产品,如 WebSphere MQ、Sonic、 Fiorano、JBossMQ 和 SpiritWave。就像 JDBC 对于数据库一样,JMS 是消息的 中间件--它使得应用程序可以通过统一的接口访问不同的消息队列产品,这个接 口提供了像 Connection 、 Topic 和 Message 这样的抽象。

许多 J2SE 应用程序使用某种形式的消息队列,但是不使用 JMS--而是使用 线程池、工作队列、任务管理器等。AWT 和 Swing 框架使用消息(事件)在模 型与视图层之间通信,JavaBean 组件利用监听器支持一种基于主题的消息。消 息队列提供了很多结构上的优点--它固有的松散耦合有利于采用灵活的、基于组 件的方法,并提供了有助于简化设计和减少互连的天然抽象边界。一个附带的好 处是,MQ 范例使得分布式的、可伸缩的和容错的设计变得更容易了,因为消息 生产者和使用者不一定需要运行在同一 JVM 中,大多数生产者/使用者任务的本 性是允许并发处理的。

J2SE 服务器应用程序的开发人员经常开发他们自己的消息层,或者从零开始 ,或者以 util.concurrent 这样的库为基础构建。通常这么做的理由是:

MQ 服务器是昂贵的和重量级的,并且由于我们不需要像持久性、分布式、事 务和验证这样更重量级的功能,构建自己内存中的消息层,只提供所需要的功能 会更容易。

虽然这在一般情况下是正确的,但是应用程序需求通常会随着时间而增加, 在开始时不需要的一些消息功能在以后可能会变得更重要了。

面向消息的应用程序的开发人员在开发过程的初期就必须做出选择--选择一 个商业消息产品,或者构建自己的更便宜的、更轻量级的解决方案。Somnifugi JMS 包结合了这两种方式--一个基于高性能的 util.concurrent 库的非持久内 存中消息队列服务,和一个符合 JMS API 的接口。与传统 JMS 提供者相比较, Somnifugi 是相当轻量级的,不管是功能上还是性能上。它只限于在一个 JVM 中使用(尽管可以取消这种限制),并缺少持久性、事务和验证功能。另一方面 ,它特别快--它比传统 JMS 实现快得多,以至于可以在因性能原因可能无法使 用消息的地方使用它。为了表明 Somnifugi 到底有多轻量级,在它的分发中包 含了几个用 JMS 主题取代 Swing/JavaBean 事件框架的例子。

增加的灵活性

Somnifugi 还提供了另一项重要的优点:现在可以开发使用 JMS 接口的组件 ,然后在部署应用程序时决定是使用更快的、内存中的 Somnifugi 提供者还是 更重量级的、但是更可靠的提供者,如 WebSphere MQ。可以将这种选择推迟到 部署时的好处非常巨大--特别是因为需求可能会在项目的开发过程中变化时--并 提供了代码重用的机会,对于自已开发的消息层来说这是不太可能做到的。

时间: 2024-10-30 05:20:56

Java理论与实践: 让J2EE脱离容器的相关文章

Java理论与实践专题

Java理论与实践: JDK 5.0中更灵活.更具可伸缩性的锁定机制 Java理论和实践: 一个有缺陷的微基准的剖析 Java理论和实践: 理解JTS ― 平衡安全性和性能 Java理论和实践: 理解JTS ― 幕后魔术 Java理论和实践: 安全构造技术 Java理论与实践: 平衡测试,第3部分:用方面检验设计约束 Java理论与实践:平衡测试,第2部分:编写和优化bug检测器 Java理论与实践:平衡测试,第1部分:不要仅编写测试,还要编写bu Java理论与实践: 您的小数点到哪里去了?

Java 理论与实践:变还是不变?

不变对象具有许多能更方便地使用它们的特性,包括不严格的同步需求和不必考虑数据讹误就能自由地共享和高速缓存对象引用.尽管不变性可能未必对于所有类都有意义,但大多数程序中至少有一些类将受益于不可变.在本月的 Java 理论与实践中,Brian Goetz 说明了不变性的一些长处和构造不变类的一些准则.请在附带的论坛中与作者和其他读者分享您关于本文的心得.(也可以单击文章顶部或底部的"讨论"来访问论坛.) 不变对象是指在实例化后其外部可见状态无法更改的对象.Java 类库中的 String.

Java 理论与实践: JDK 5.0 中更灵活、更具可伸缩性的锁定机制

伸缩 内容: synchronized 快速回顾 对 synchronized 的改进 比较 ReentrantLock 和 synchronized 的可伸缩性 条件变量 这不公平 结束语 参考资料 关于作者 对本文的评价 相关内容: Java 理论与实践 系列 Synchronization is not the enemy Reducing contention IBM developer kits for the Java platform (downloads) 订阅: develop

Java理论与实践: 您的小数点到哪里去了?

许多程序员在其整个开发生涯中都不曾使用定点或浮点数,可能的例外是, 偶尔在计时测试或基准测试程序中会用到.Java语言和类库支持两类非整数类型 ― IEEE 754 浮点( float 和 double ,包装类(wrapper class)为 Float 和 Double ),以及任意精度的小数( java.math.BigDecimal ).在本月的 Java 理论和实践中,Brian Goetz 探讨了在 Java 程序中使用非整数类型时一 些常碰到的陷阱和"gotcha". 虽

Java理论与实践: 垃圾收集简史

Java 语言可能是使用最广泛的依赖于垃圾收集的编程语言,但是它并不是第 一个.垃圾收集已经成为了包括 Lisp.Smalltalk.Eiffel.Haskell.ML. Scheme和 Modula-3 在内的许多编程语言的一个集成部分,并且从 20 世纪 60 年代早期就开始使用了.在 Java 理论与实践的本篇文章中,Brian Goetz 描述 了垃圾收集最常用的技术. 垃圾收集的好处是无可争辩的 ―― 可靠性提高.使内存管理与类接口设计 分离,并使开发者减少了跟踪内存管理错误的时间.著

Java理论与实践: 构建一个更好的HashMap

ConcurrentHashMap 是 Doug Lea的 util.concurrent 包的一部分,它提供 比Hashtable 或者 synchronizedMap 更高程度的并发性.而且,对于大多数成 功的 get() 操作它会设法避免完全锁定,其结果就是使得并发应用程序有着非 常好的吞吐量.这个月,BrianGoetz 仔细分析了 ConcurrentHashMap的代码, 并探讨 Doug Lea 是如何在不损失线程安全的情况下取得这么骄人成绩的. 在7月份的那期 Java理论与实践

Java理论与实践: 消除bug

很多有关编程风格的建议都是为了创建高质量.可维护的代码,这很合理, 因为最容易修复 bug 的时间就是在产生 bug 之前(少量的预防措施--).遗 憾的是,只预防往往是不够的,虽然有一些精巧的工具可以帮助您创建好的代码 ,但是很少有工具可以帮助您分析.维护或提高现有代码的质量. 写线程安全的类很难,而分析现有类的线程安全性更难,增强类使其仍然保 持线程安全也很难.以隐含假定.不变式以及预期用例(虽然在开发人员的头脑 中很清晰,但是没有以设计笔记.注释或者文档的方式记录下来)的方式编写完 类之后

Java理论与实践: 修复Java内存模型,第1部分

活跃了将近三年的 JSR 133,近期发布了关于如何修复 Java 内存模型 (Java Memory Model, JMM)的公开建议.原始 JMM 中有几个严重缺陷,这导 致了一些难度高得惊人的概念语义,这些概念原来被认为很简单,如 volatile .final 以及 synchronized.在这一期的 Java 理论与实践 中,Brian Goetz 展示了如何加强 volatile 和 final 的语义,以修复 JMM.这些更改有些已经 集成在 JDK 1.4 中:而另一些将会包含

Java理论与实践:做个好的(事件)侦听器

观察者模式在 Swing 开发中很常见,在 GUI 应用程序以外的场景中,它对 于消除组件的耦合性也非常有用.但是,仍然存在一些侦听器登记和调用方面的 常见缺陷.在 Java 理论与实践 的这一期中,Java 专家 Brian Goetz 就如何 做一个好的侦听器,以及如何对您的侦听器也友好,提供了一些感觉很好的建议 .请在相应的 讨论论坛 上与作者和其他读者分享您对这篇文章的想法.(您也 可以单击本文顶部或底部的 讨论 访问论坛.) Swing 框架以事件侦听器的形式广泛利用了观察者模式(也称