Java服務定位器模式

當我們想要使用JNDI查找來定位各種服務時,使用服務定位器設計模式。 考慮到爲服務查找JNDI的高成本,所以在服務定位器模式使用緩存技術。 首次需要服務時,服務定位器在JNDI中查找並緩存服務對象。 通過服務定位器進一步查找或相同的服務在其緩存中完成,這在很大程度上提高了應用的性能。 以下是這種類型的設計模式的實體。

  • 服務 - 將處理請求的實際服務。 將在JNDI服務器中查找此類服務的引用。
  • 上下文/初始上下文 - JNDI上下文攜帶對用於查找目的的服務的引用。
  • 服務定位器 - 服務定位器是通過JNDI查找緩存服務獲得服務的單一聯繫點。
  • 緩存 - 用於存儲服務的引用以重用它們的緩存。
  • 客戶端 - 客戶端是通過ServiceLocator調用服務的對象。

實現實例

在這個實現的實例中,將創建一個ServiceLocatorInitialContextCacheService作爲表示實體的各種對象。 Service1Service2用來表示具體服務。

ServiceLocatorPatternDemo是一個演示類,在這裏充當客戶端,將使用ServiceLocator演示服務定位器設計模式。

服務定位器模式示例的結構如下圖所示 -

Java服務定位器模式

第1步

創建一個名爲 Service 的接口,其代碼如下 -
Service.java

public interface Service {
   public String getName();
   public void execute();
}

第2步

創建具體的服務類,其代碼如下 -

Service1.java

public class Service1 implements Service {
   public void execute(){
      System.out.println("Executing Service1");
   }

   @Override
   public String getName() {
      return "Service1";
   }
}

Service2.java

public class Service2 implements Service {
   public void execute(){
      System.out.println("Executing Service2");
   }

   @Override
   public String getName() {
      return "Service2";
   }
}

第3步

JNDI查找創建InitialContext 類,其代碼如下 -
InitialContext.java

public class InitialContext {
   public Object lookup(String jndiName){

      if(jndiName.equalsIgnoreCase("SERVICE1")){
         System.out.println("Looking up and creating a new Service1 object");
         return new Service1();
      }
      else if (jndiName.equalsIgnoreCase("SERVICE2")){
         System.out.println("Looking up and creating a new Service2 object");
         return new Service2();
      }
      return null;        
   }
}

第4步

創建緩存類,其代碼如下 -
Cache.java

import java.util.ArrayList;
import java.util.List;

public class Cache {

   private List<Service> services;

   public Cache(){
      services = new ArrayList<Service>();
   }

   public Service getService(String serviceName){

      for (Service service : services) {
         if(service.getName().equalsIgnoreCase(serviceName)){
            System.out.println("Returning cached  " + serviceName + " object");
            return service;
         }
      }
      return null;
   }

   public void addService(Service newService){
      boolean exists = false;

      for (Service service : services) {
         if(service.getName().equalsIgnoreCase(newService.getName())){
            exists = true;
         }
      }
      if(!exists){
         services.add(newService);
      }
   }
}

第5步

創建服務定位器,其代碼如下 -

ServiceLocator.java

public class ServiceLocator {
   private static Cache cache;

   static {
      cache = new Cache();        
   }

   public static Service getService(String jndiName){

      Service service = cache.getService(jndiName);

      if(service != null){
         return service;
      }

      InitialContext context = new InitialContext();
      Service service1 = (Service)context.lookup(jndiName);
      cache.addService(service1);
      return service1;
   }
}

第6步

使用ServiceLocator類來演示服務定位器設計模式。
ServiceLocatorPatternDemo.java

public class ServiceLocatorPatternDemo {
   public static void main(String[] args) {
      Service service = ServiceLocator.getService("Service1");
      service.execute();
      service = ServiceLocator.getService("Service2");
      service.execute();
      service = ServiceLocator.getService("Service1");
      service.execute();
      service = ServiceLocator.getService("Service2");
      service.execute();        
   }
}

第7步

驗證輸出,執行上面的代碼得到以下結果 -

Looking up and creating a new Service1 object
Executing Service1
Looking up and creating a new Service2 object
Executing Service2
Returning cached  Service1 object
Executing Service1
Returning cached  Service2 object
Executing Service2