网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
Firefox | IE | Maxthon | 迅雷 | 电驴 | BitComet | FlashGet | QQ | QQ空间 | Vista | 输入法 | Ghost | Word | Excel | wps | Powerpoint
asp | .net | php | jsp | Sql | c# | Ajax | xml | Dreamweaver | FrontPages | Javascript | css | photoshop | fireworks | Flash | Cad | Discuz!
当前位置 > 网站建设学院 > 网络编程 > Java
Tag:注入,存储过程,分页,安全,优化,xmlhttp,fso,jmail,application,session,防盗链,stream,无组件,组件,md5,乱码,缓存,加密,验证码,算法,cookies,ubb,正则表达式,水印,索引,日志,压缩,base64,url重写,上传,控件,Web.config,JDBC,函数,内存,PDF,迁移,结构,破解,编译,配置,进程,分词,IIS,Apache,Tomcat,phpmyadmin,Gzip,触发器,socket
网络编程:ASP教程,ASP.NET教程,PHP教程,JSP教程,C#教程,数据库,XML教程,Ajax,Java,Perl,Shell,VB教程,Delphi,C/C++教程,软件工程,J2EE/J2ME,移动开发
本月文章推荐
.软件开发的“简单”艺术—有感于.
.Drive 对象.
.Java实现类MSN、QQ好友上线通知界.
.Java理论和实践:用软引用阻止内存.
.GEF-SDK-3.0.
.struts构建文件上传(5).
.JAVA数组和JAVA集合类的对象操作.
.[JAVA100例]037、读取随机文件.
.使用JavaExcel API 往excel中插入.
.Java核心代码例程之:CalendarDe.
.Eclipse 3.1 M3 快捷键.
.到2004年Java技术发展预测.
.Java编程新手入门:运算符优先级.
.Java 数据类型 和 与String之间的.
.Sun.XVR-1200图形加速器的使用.
.EJB轻松进阶(4).
.轻量级IoC容器来扩展ANT享受SPRI.
.使用CMP2和XML处理动态数值对象.
.戏说java设计模式中的命令模式.
.开发MIDP联网应用程序.

EJB 最佳实践:工业强度的 JNDI 优化

发表日期:2008-1-5



  Brett McLaughlin 在这篇 EJB 最佳实践专栏文章中研究了 JNDI 查找,它是几乎所有的 EJB 交互中不可或缺并且常见的部分。遗憾的是,JNDI 操作几乎总是需要性能开销。在本技巧文章中,Brett 向您展示了 home 接口工厂是如何降低您 EJB 应用程序中 JNDI 查找开销的。
  每种 EJB 组件(会话、实体和消息驱动的)都有 home 接口。home 接口是 bean 的操作基础;一旦您找到它,就可以使用该 bean 的功能。EJB 应用程序依靠 JNDI 查找来访问其 bean 的 home 接口。因为 EJB 应用程序往往运行多个 bean,并且因为许多组件中经常使用 JNDI 查找,所以应用程序大部分性能开销都花费在这些查找上。
  
  在这篇技巧文章中,我们将研究一些最常用的 JNDI 优化。非凡地,我们将向您展示如何将高速缓存和通用助手类组合使用,以创建针对 JNDI 开销的工厂风格的解决方案。
  
  减少上下文实例
  清单 1 显示了一段典型的 EJB 代码,它需要多次 JNDI 查找。请花一点时间研究代码,然后我们将对它进行优化以获得更佳性能。
  
  清单 1. 典型的 EJB 查找
  public boolean buyItems(PaymentInfo paymentInfo, String storeName,
  List items) {
   // Load up the initial context
   Context ctx = new InitialContext();
  
   // Look up a bean's home interface
   Object obj = ctx.lookup("Java:comp/env/ejb/PurchaseHome");
   PurchaseHome purchaseHome =
   (PurchaseHome)PortableRemoteObject.narrow(obj, PurchaseHome.class);
   Purchase purchase = purchaseHome.create(paymentInfo);
  
   // Work on the bean
   for (Iterator i = items.iterator(); i.hasNext(); ) {
   purchase.addItem((Item)i.next());
   }
  
   // Look up another bean
   Object obj = ctx.lookup("java:comp/env/ejb/InventoryHome");
   InventoryHome inventoryHome =
   (InventoryHome)PortableRemoteObject.narrow(obj, InventoryHome.class);
   Inventory inventory = inventoryHome.findByStoreName(storeName);
  
   // Work on the bean
   for (Iterator i = items.iterator(); i.hasNext(); )
   inventory.markAsSold((Item)i.next());
   }
  
   // Do some other stuff
  }
  
  
  
  
  尽管这个示例多少有点刻意,但它确实揭示了使用 JNDI 时的一些最明显的问题。对于初学者,您应该问问自己,新建 InitialContext 对象是否必需。很可能在应用程序代码的其它地方已经装入了这个上下文,而我们又在这里创建了一个新的。高速缓存 InitialContext 实例会立即促使性能提高,如清单 2 所示:
  
  清单 2. 高速缓存 InitialContext 实例
  public static Context getInitialContext() {
   if (initialContext == null) {
   initialContext = new InitialContext();
   }
  
   return initialContext;
  }
  
  
  
  
  通过对 getInitialContext() 使用助手类,而不是为每个操作都实例化一个新的 InitialContext,我们将遍布在应用程序中的上下文数量减少为一个。
  
  哦 — 线程化会怎么样?
  假如您对此处提出的解决方案的线程化感到担心,那大可不必。两个线程同时进行 getInitialContext() 是绝对有可能的(从而一次创建两个上下文),但只有首次调用该方法时才会发生此类错误。因为问题至多只会发生一次,所以同步是不必要的,实际上,同步引入的复杂性比它所解决的复杂性更多。
  
  优化查找
  高速缓存上下文实例这个步骤的方向是正确的,但仅这样做,还不足以完成优化。我们每次调用 lookup() 方法时都会执行一次新查找,并返回 bean 的 home 接口的新实例。至少,JNDI 查找通常是这样编码的。但假如每个 bean 都只有一个 home 接口,并在多个组件上共享这个接口,这样不是更好吗?
  
  我们可以高速缓存每个单独的 bean 引用,而不是反复查找 PurchaseHome 或 InventoryHome 的 home 接口;这是一种解决方案。但我们真正想要的是一种更通用的机制:在 EJB 应用程序中高速缓存 home 接口。
  
  答案是创建通用助手类,它既可以为应用程序中的每个 bean 获取初始上下文,又可以为它们查找 home 接口。此外,这个类还应该能够为各种应用程序组件治理每个 bean 的上下文。清单 3 中所示的通用助手类将充当 EJB home 接口的工厂:
  
  清单 3. EJB home 接口工厂
  package com.ibm.ejb;
  
  import java.util.Map;
  import javax.ejb.EJBHome;
  import javax.naming.Context;
  import javax.naming.InitialContext;
  import javax.naming.NamingException;
  
  public class EJBHomeFactory {
  
   private static EJBHomeFactory;
  
   private Map homeInterfaces;
   private Context context;
  
   // This is private, and can't be instantiated directly
   private EJBHomeFactory() throws NamingException {
   homeInterfaces = new HashMap();
  
   // Get the context for caching purposes
   context = new InitialContext();
  
   /**
   * In non-J2EE applications, you might need to load up
   * a properties file and get this context manually. I've
   * kept this simple for demonstration purposes.
   */
   }
  
   public static EJBHomeFactory getInstance() throws NamingException {
   // Not completely thread-safe, but good enough
   // (see note in article)
   if (instance == null) {
   instance = new EJBHomeFactory();
   }
   return instance;
   }
  
   public EJBHome lookup(String jndiName, Class homeInterfaceClass)
   throws NamingException {
  
   // See if we already have this interface cached
   EJBHome homeInterface = (EJBHome)homeInterfaces.get(homeClass);
  
   // If not, look up with the supplied JNDI name
   if (homeInterface == null) {
   Object obj = context.lookup(jndiName);
   homeInterface =
   (EJBHome)PortableRemoteObject.narrow(obj, homeInterfaceClass);
  
   // If this is a new ref, save for caching purposes
   homeInterfaces.put(homeInterfaceClass, homeInterface);
   }
   return homeInterface;
   }
  }
  
  
  
  
  EJBHomeFactory 类内幕
  home 接口工厂的要害在 homeInterfaces 映射中。该映射存储了供使用的每个 bean 的 home 接口;这样,home 接口实例可以反复使用。您还应注重,映射中的要害并不是传递到 lookup() 方法的 JNDI 名称。将同一 home 接口绑定到不同 JNDI 名称是很常见的,但这样做会在您的映射中产生副本。通过依靠类本身,您就可以确保最终不会为同一个 bean 创建多个 home 接口。
  
  将新的 home 接口工厂类插入清单 1 的原始代码,这样将会产生优化的 EJB 查找,如清单 4 所示:
  
  清单 4. 改进的 EJB 查找
  public boolean buyItems(PaymentInfo paymentInfo, String storeName,
  List items) {
  
   EJBHomeFactory f = EJBHomeFactory.getInstance();
  
   PurchaseHome purchaseHome =
   (PurchaseHome)f.lookup("java:comp/env/ejb/PurchaseHome",
   PurchaseHome.class);
   Purchase purchase = purchaseHome.create(paymentInfo);
  
   // Work on the bean
   for (Iterator i = items.iterator(); i.hasNext(); ) {
   purchase.addItem((Item)i.next());
   }
  
   InventoryHome inventoryHome =
   (InventoryHome)f.lookup("java:comp/env/ejb/InventoryHome",
   InventoryHome.class);
   Inventory inventory = inventoryHome.findByStoreName(storeName);
  
   // Work on the bean
   for (Iterator i = items.iterator(); i.hasNext(); ) {
   inventory.markAsSold((Item)i.next());
   }
  
   // Do some other stuff
  }
  
  
  
  
  随着时间的推进,除了更清楚之外(至少按我的观点),以上工厂优化的 EJB 查找将执行得更快。您第一次使用这个新类时,将花费所有正常查找开销(假定应用程序的其它部分没有付出过这种开销),但将来的所有
上一篇:布署和使用 Enterprise JavaBeans 组件 人气:813
下一篇:企业级EJB开发实例 人气:646
浏览全部Java的内容 Dreamweaver插件下载 网页广告代码 祝你圣诞节快乐 2009年新年快乐