用SERVICE LOCATOR 模式实现命名访问

原创 2003年09月10日 11:26:00

SERVICE LOCATOR 模式实现命名访问服务

 <?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />

B/S开发中, 我们经常要用到名称服务,如JNDIXMLNS等。名称服务随不同厂家而不同。每次需要获得名称服务时,需要适当的名称环境信息,然后查找服务,重复查找的成本很高。

 

此外,在持久性框架中,要求将所有的服务访问都包装到对象中,开发人员不需要知道名称服务后面的平台(数据库)类型,及任何安全信息或地址。在一个大型软件产品间存在多个EJB和多个数据源连接时(我们目前只有一个写死的数据源WEBGL)需要一个类来实现统一的访问管理。

 

因此,这就要求我们对这些访问进行封装隔离。这时我们就可以利用SERVICE LOCATOR模式。

 

 

我们将利用一个文件service.properties来管理所有的命名服务。例子代码实现了EJB本地,数据源的访问。

格式如下:

DEFAULTDS=webgl

NTCLASSREF=NTHome.class

 

把它保存在CLASSPATH引用的路径里。

 

源代码:

 

Package  com.learn;

 

import java.util.Hashtable;

import java.util.Properties;

import java.io.*;

import javax.ejb.EJBHome;

import javax.naming.InitialContext;

import javax.naming.Context;

import javax.naming.NamingException;

import javax.rmi.PortableRemoteObject;

import java.sql.Connection;

import java.sql.SQLException;

import javax.sql.DataSource;

 

public class ServiceLocator {

  private static ServiceLocator serviceLocatorRef = null;

  private static Hashtable ejbHomeCache = null;

  private static Hashtable dataSourceCache = null;

  private static Properties serviceCache = null;

  static {

    serviceLocatorRef = new ServiceLocator();

  }

 

  private ServiceLocator() {

    ejbHomeCache = new Hashtable();

    dataSourceCache = new Hashtable();

    try {

      String serviceFileName = "service.properties";

      serviceCache.load(new FileInputStream(serviceFileName));

    }

    catch (IOException e) {

      System.out.println(e.toString());

    }

  }

 

  /**

   * 使用singleton.模式静态对象多次调用节省资源   */

 

  public static ServiceLocator getInstance() {

    return serviceLocatorRef;

  }

 

  /*

   * 由键获得键值,一般在数据源,XMLNS访问用到这个方法

   */

  static private String getServiceName(String serviceId)

    throws ServiceLocatorException{

    String serviceName=null;

    if (serviceCache.containsKey(serviceId)) {

      serviceName = (String) serviceCache.get(serviceId);

    }

    else {

      throw new ServiceLocatorException(

        "Unable to locate the service statement requested");

    }

    return serviceName;

  }

 

/*************************************************

 * EJB本地类引用

*************************************************/  

  static private Class getEJBHomeRef(String serviceId) throws

    ServiceLocatorException {

    Class homeRef = null;

    if (serviceCache.containsKey(serviceId)) {

      homeRef = (Class) serviceCache.get(serviceId);

    }

    else {

      throw new ServiceLocatorException(

        "Unable to locate the service statement requested");

    }

    return homeRef;

  }

 

/************************************************************************

 * 获得EJBHome对象

 ***********************************************************************/  

  public EJBHome getEJBHome(String serviceId) throws ServiceLocatorException {

    EJBHome ejbHome = null;

    try {

      //先检查缓存是否存在EJBHome接口

      if (ejbHomeCache.containsKey(serviceId)) {

        ejbHome = (EJBHome) ejbHomeCache.get(serviceId);

        return ejbHome;

      }

      else {

        //如果没有存在,则解析并存到缓存中

        Context ctx = new InitialContext();

        Object jndiRef = ctx.lookup(serviceId);

        Object portableObj = PortableRemoteObject.narrow(jndiRef,

          getEJBHomeRef(serviceId));

        ejbHome = (EJBHome) portableObj;

        ejbHomeCache.put(serviceId, ejbHome);

        return ejbHome;

      }

    }

    catch (NamingException e) {

      throw new ServiceLocatorException(

        "Naming exception error in ServiceLocator.getEJBHome()", e);

    }

  }

 

  /*

   * 获得JNDI数据源

   */

  public Connection getDBConn(String serviceId) throws

    ServiceLocatorException {

    Connection conn = null;

    String serviceName=getServiceName(serviceId);

    try {

      /*Checking to see if the requested DataSource is in the Cache*/

      if (dataSourceCache.containsKey(serviceId)) {

        DataSource ds = (DataSource) dataSourceCache.get(serviceId);

        conn = ( (DataSource) ds).getConnection();

        return conn;

      }

      else {

        /*

         * The DataSource was not in the cache.  Retrieve it from JNDI

         * and put it in the cache.

         */

        Context ctx = new InitialContext();

        DataSource newDataSource = (DataSource) ctx.lookup(serviceName);

        dataSourceCache.put(serviceId, newDataSource);

        conn = newDataSource.getConnection();

        return conn;

      }

    }

    catch (SQLException e) {

      throw new ServiceLocatorException("A SQL error has occurred in " +

                                        "ServiceLocator.getDBConn()", e);

    }

    catch (NamingException e) {

      throw new ServiceLocatorException("A JNDI Naming exception has occurred " +

                                        " in ServiceLocator.getDBConn()", e);

    }

    catch (Exception e) {

      throw new ServiceLocatorException("An exception has occurred " +

                                        " in ServiceLocator.getDBConn()", e);

    }

  }

 

}

 

 

异常处理类:

 

package com.learn;

public class ServiceLocatorException extends DataAccessException{

  public ServiceLocatorException(String pExceptionMsg){

    super(pExceptionMsg);

  }

 

  public ServiceLocatorException(String pExceptionMsg, Throwable pException){

    super(pExceptionMsg, pException);

  }

 

}

 

 

参考书籍:

        《实用J2EE设计模式编程指南》

              WEB服务精髓》

         J2EE BEA WEBLOGIC SERVER

 

 

请大家指教:lisong@anyi.com.cn

用SERVICE LOCATOR 模式实现访问命名服务

      在B/S开发中, 我们经常要用到名称服务,如JNDI,XMLNS等。名称服务随不同厂家而不同。每次需要获得名称服务时,需要适当的名称环境信息,然后查找服务,重复查找的成本很高。此外,在持久...
  • touchSea
  • touchSea
  • 2006年05月20日 21:15
  • 1059

用SERVICE LOCATOR 模式实现访问命名服务

google_ad_client = "pub-8800625213955058";/* 336x280, 创建于 07-11-21 */google_ad_slot = "0989131976";...
  • java169
  • java169
  • 2008年05月24日 02:35
  • 156

服务定位器模式(Service Locator Pattern)详解和代码示范

模式所要解决的问题Service Locator模式想要解决的问题是解耦合服务提供者和用户,用户无需直接访问具体的服务提供者类。比如发送短信/邮件,在应用程序的很多地方都会被使用,有两种简单的方法来实...
  • iefreer
  • iefreer
  • 2013年07月16日 14:39
  • 9383

设计模式学习—服务定位模式(Service Locator Design Pattern)

设计模式之服务定位模式,一种JavaEE模式
  • bird3014
  • bird3014
  • 2017年09月08日 17:52
  • 487

设计模式【服务器定位模式Service Locator Pattern】

服务定位器模式 服务定位器模式(Service Locator Pattern)用在我们想使用 JNDI 查询定位各种服务的时候。考虑到为某个服务查找 JNDI 的代价很高,服务定位器模式充分利...
  • qiaoquan3
  • qiaoquan3
  • 2017年11月25日 11:59
  • 81

服务定位器模式(service locator)

Using a Service Locator使用服务定位器    The key benefit of a Dependency Injector is that it removes the de...
  • cz_hyf
  • cz_hyf
  • 2007年02月10日 20:20
  • 10398

Service Locator 模式 的学习

这两天有时间对使用多年的mvc4架构的代码仔细看了看。当时因为以学习为目的向师兄借用的架构,虽然时隔多年,有很多设计理念还是值得回味! Service Locator 模式 什么是Servi...
  • hb0746
  • hb0746
  • 2016年04月11日 17:54
  • 782

使用service locator快速定位你的服务

申明:本人对java了解不太深,发表的观点可能有不当之处,如有之请不吝赐教。service locator模式在老早的j2ee中就被提出来,原因是在一个程序中有大量服务(service)时,使用者调用...
  • googlezhang
  • googlezhang
  • 2011年04月05日 13:27
  • 1379

Java Service Locator Pattern(服务器定位模式)

Java Service Locator Pattern(服务器定位模式) 服务定位器模式(Service Locator Pattern)用在我们想使用 JNDI 查询定位各种服务的时候。考虑到为某...
  • mazaiting
  • mazaiting
  • 2017年09月29日 01:08
  • 45

【设计模式】服务定位器模式(Service Locator Pattern)

服务定位器模式 服务定位器模式(Service Locator Pattern)用在我们想使用 JNDI 查询定位各种服务的时候。考虑到为某个服务查找 JNDI 的代价很高,服务定位器模式充分利...
  • MR_XuZhe
  • MR_XuZhe
  • 2017年12月01日 08:51
  • 104
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:用SERVICE LOCATOR 模式实现命名访问
举报原因:
原因补充:

(最多只允许输入30个字)