http://wxg6203.iteye.com/blog/680830

最近因为工作需要开始学习Ejb3,遇到了一个让我很郁闷的事情,做一下小小的总结——小心new InitialContext()。

在做客户端的时候,发现连接服务器,搜索数据库,然后返回结果集。每一次执行的时候,第一次总要花更多的时间,之后每一次操作时间就要快很多了。期间找了很多方法,都行不通。一开始以为是Ejb服务器建立服务消耗时间,后来觉得不对,因为Jboss启动的时候,已经将服务启动了。经过一周的排查,终于发现原来是因为new InitialContext()消耗了大量的时间,之后的lookup()方法也会消耗一定的时间。其中,在网络状态良好的情况下,每一次new InitialContext()方法花费大概100毫秒到200毫秒之间,而每一次lookup()大概要花10毫秒到30毫秒之间。因此,决定对代码进行优化,创建了EJBHomeFactory工具类,使用到了单例模式,欢迎大家指教。以下为该类代码:

Java代码 
import javax.naming.InitialContext;   
import javax.naming.NamingException;   
  
import com.cithinc.util.Tool;   
  
public class EJBHomeFactory {   
    private static EJBHomeFactory instance;   
    private InitialContext context;   
  
    private EJBHomeFactory() throws NamingException {   
        context = Tool.getInitialContext();   
    }   
  
    public static EJBHomeFactory getInstance() throws NamingException {   
        if (instance == null) {   
            instance = new EJBHomeFactory();   
        }   
        return instance;   
    }   
  
    public Object lookup(String jndiName) throws NamingException {   
        Object obj = new Object();   
        obj = context.lookup(jndiName);   
        return obj;   
    }   
}

import javax.naming.InitialContext; 
import javax.naming.NamingException;

import com.cithinc.util.Tool;

public class EJBHomeFactory { 
private static EJBHomeFactory instance; 
private InitialContext context;

private EJBHomeFactory() throws NamingException { 
context = Tool.getInitialContext(); 
}

public static EJBHomeFactory getInstance() throws NamingException { 
if (instance == null) { 
instance = new EJBHomeFactory(); 

return instance; 
}

public Object lookup(String jndiName) throws NamingException { 
Object obj = new Object(); 
obj = context.lookup(jndiName); 
return obj; 

}

其中,Tool.java的文件内容如下:

Java代码 
import java.util.Hashtable;   
  
import javax.naming.Context;   
import javax.naming.InitialContext;   
import javax.naming.NamingException;   
  
public class Tool {   
    @SuppressWarnings("unchecked")   
    public static InitialContext getInitialContext() throws NamingException {   
        Hashtable environment = new Hashtable();   
        environment.put(Context.INITIAL_CONTEXT_FACTORY,"org.jnp.interfaces.NamingContextFactory");   
        environment.put(Context.URL_PKG_PREFIXES,"org.jboss.naming:org.jnp.interfaces");   
        environment.put(Context.PROVIDER_URL, "jnp://127.0.0.1:1099");   
        return new InitialContext(environment);   
    }   
}

import java.util.Hashtable;

import javax.naming.Context; 
import javax.naming.InitialContext; 
import javax.naming.NamingException;

public class Tool { 
@SuppressWarnings("unchecked") 
public static InitialContext getInitialContext() throws NamingException { 
Hashtable environment = new Hashtable(); 
environment.put(Context.INITIAL_CONTEXT_FACTORY,"org.jnp.interfaces.NamingContextFactory"); 
environment.put(Context.URL_PKG_PREFIXES,"org.jboss.naming:org.jnp.interfaces"); 
environment.put(Context.PROVIDER_URL, "jnp://127.0.0.1:1099"); 
return new InitialContext(environment); 

}

然后这样调用:

Java代码 
EJBHomeFactory f = EJBHomeFactory.getInstance();   
Object o = f.lookup(remote);

EJBHomeFactory f = EJBHomeFactory.getInstance(); 
Object o = f.lookup(remote);

这样就可以保证只初始化一次上下文实例,节省大量的时间。

分享到:  
评论
2 楼 surpassno 2013-08-29  
1 楼 ybzshizds 2010-08-06  
这种用工厂模式解决的办法其实还是有一个问题,就是ejb容器重启后,客户端应该也要重新去new InitialContext,否则用原来的InitialContext去lookup,会抛出错。

想想后,我是这样去解决的。

为每个ejbhome,新建一个ejbHelper类

如:

  1. package com.company.vas.ejb.helper;
  2. import java.rmi.RemoteException;
  3. import javax.ejb.CreateException;
  4. import javax.naming.NamingException;
  5. import com.company.util.Log;
  6. import com.company.vas.ejb.Invoice;
  7. import com.company.vas.ejb.home.InvoiceHome;
  8. public class InvoiceHelper {
  9. private static final String CLASS_NAME = "InvoiceHelpler";
  10. private static InvoiceHome home;
  11. public static Invoice getInvoice() {
  12. try {
  13. if (home == null) {
  14. home = (InvoiceHome) EjbGetter.getEJBHome(
  15. IInvoice.JNDI_NAME, InvoiceHome.class);
  16. }
  17. return home.create();
  18. } catch (NamingException e) {
  19. Log.error(CLASS_NAME, "getInvoice()", e.getMessage());
  20. home = null;
  21. } catch (RemoteException e) {
  22. Log.error(CLASS_NAME, "getInvoice()", e.getMessage());
  23. home = null;
  24. } catch (CreateException e) {
  25. Log.error(CLASS_NAME, "getInvoice()", e.getMessage());
  26. home = null;
  27. }
  28. return null;
  29. }
  30. }
  31. package com.company.vas.ejb.helper;
  32. import java.util.Properties;
  33. import javax.ejb.EJBHome;
  34. import javax.naming.Context;
  35. import javax.naming.InitialContext;
  36. import javax.naming.NamingException;
  37. public class EjbGetter {
  38. public static EJBHome getEJBHome(String service_jndiname, Class homeInterface) throws NamingException{
  39. Properties env = new Properties();
  40. env.put(Context.INITIAL_CONTEXT_FACTORY, "org.jnp.interfaces.NamingContextFactory");
  41. env.put(Context.PROVIDER_URL, "192.168.60.120:1099");
  42. env.put("java.naming.factory.url.pkgs", "org.jboss.naming:org.jnp.interfaces");
  43. Context ic = new InitialContext(env);
  44. EJBHome ejbHome = (EJBHome)javax.rmi.PortableRemoteObject.narrow(ic.lookup(service_jndiname), homeInterface);
  45. return ejbHome;
  46. }
  47. }

InitialContext和lookup的更多相关文章

  1. InitialContext和lookup(转)

    原文地址:http://wxg6203.iteye.com/blog/680830 最近因为工作需要开始学习Ejb3,遇到了一个让我很郁闷的事情,做一下小小的总结——小心new InitialCont ...

  2. InitialContext与lookup

    Context initial = new InitialContext(); Object objref = initial.lookup("java:comp/env/ejb/Simpl ...

  3. Jboss7.1 local EJB lookup problem

    We are trying to lookup for an Local EJB in JBoss7.1, but we get an ClassCast Exception. This local ...

  4. Class loading in JBoss AS 7--官方文档

    Class loading in AS7 is considerably different to previous versions of JBoss AS. Class loading is ba ...

  5. 浅谈JNDI的使用

    原文:http://www.weicoop.com/web/article/52.html 关于什么是JNDI的概念这里不做解释,本文作为初学者根据所了解到内容做些总结,主要内容如下: 1.JNDI使 ...

  6. JNDI support differences between Tibco EMS and ActiveMQ

    Introduction Recently our team was working on Veracity Quick Start sprint, when I was trying to migr ...

  7. 再说JNDI

    说到JNDI,即熟悉又陌生,熟悉在常常使用,如EJB3.0中的@EJB注入,底层实现即是JNDI的方式:喜闻乐见的: Context ctx=new InitialContext(); Object ...

  8. java_web学习(12)JDBC

    数据持久化       持久化(persistence):把数据保存到可掉电式存储设备中以供之后使用.大多数情况下,特别是企业级应用,数据持久化意味着将内存中的数据保存到硬盘上加以”固化”,而持久化的 ...

  9. Java命名和目录接口——JNDI

    JNDI即Java命名和目录接口(JavaNaming and Directory Interface),它属于J2EE规范范畴,是J2EE的核心技术之一,提供了一组接口.类和关于命名空间的概念.JD ...

随机推荐

  1. 为Chrome多账户添加单独的快捷方式

    Chrome的多账户功能非常好用,每个账户都有自己的独立的收藏夹.个人设置等.但是,当你要使用的账户不是默认账户时,必须经过一个切换的操作.本文将简单的介绍一个如何各账户添加快捷方式,从而实现直接登陆 ...

  2. linux之网络配置相关

    ubuntu的网络配置文件在 /etc/network/intrfaces; suse的网络配置在          /etc/sysconfig/network/下面,每个网卡一个配置文件. int ...

  3. httpd配置文件中重写函数Rewrite

    [RewriteCond%{HTTP_HOST}^(www\.)?xxx\.com$] 这是重写条件,前面%{HTTP_HOST}表示当前访问的网址,只是指前缀部分,格式是www.xxx.com不包括 ...

  4. 报错kernel:NMI watchdog: BUG: soft lockup - CPU#0 stuck for 26s

    近期在服务器跑大量高负载程序,造成cpu soft lockup.如果确认不是软件的问题. 解决办法: #追加到配置文件中 echo 30 > /proc/sys/kernel/watchdog ...

  5. Lock flag DX

    https://msdn.microsoft.com/en-us/library/windows/desktop/bb322846(v=vs.85).aspx discard nooverwrite ...

  6. Vue自定义指令和路由

    一.自定义指令 除了默认设置的核心指令( v-model 和 v-show ), Vue 也允许注册自定义指令. 下面我们注册一个全局指令 v-focus,该指令的功能是在页面加载时,元素获得焦点: ...

  7. 如何在阿里云服务器搭建FTP服务器,在本地电脑连接并操作

    首先你需要有一个阿里云的ECS服务器 并且开通了公网宽带(话说也不贵,开来玩玩还是可以的,第一次买会比较便宜,第二次买1M的宽带两天是九毛多吧~) 开通了宽带之后,ECS服务器就可以上网了 如果嫌弃阿 ...

  8. 2017.7.31 ELK+logback+redis的使用

    参考来自:spring mvc+ELK从头开始搭建日志平台 0 前提 ELK安装成功 redis安装成功 使用logback的项目运行成功 1 配置文件 1.1 pom.xml 为了使用logback ...

  9. eCos系统无法正确链接到在C++源文件里自己定义的cyg_user_start函数的问题和解决的方法

    在C++源文件里定义cyg_user_start函数前,将其声明成C函数.就可以解决这个问题. eCos官网:http://ecos.sourceware.org eCos中文技术网:http://w ...

  10. 使用Junit4对web项目进行测试(一)Junit初配置

    Junit测试用例不是用来证明你是对的,而是用来证明你没有错 1.功能   -在项目未在浏览器运行之前对获得的结果和预期的结果进行比较调试,减少BUG和发布时的修复工作 2.测试类和代码类应分开存放. ...