服務定位器模式


服務定位器模式(Service Locator Pattern)用在我們想要使用 JNDI 查詢定位各種服務的時候。考慮到為某項服務尋找 JNDI 的代價很高,服務定位器模式充分利用了快取技術。在首次要求某個服務時,服務定位器會在 JNDI 中尋找服務,並快取該服務物件。當再次請求相同的服務時,服務定位器會在它的快取中查找,這可以在很大程度上提高應用程式的效能。以下是這種設計模式的實體。

  • 服務(Service) - 實際處理請求的服務。對這種服務的引用可以在 JNDI 伺服器中查找到。

  • Context / 初始的 Context - JNDI Context 帶有對要尋找的服務的參考。

  • 服務定位器(Service Locator) - 服務定位器是透過 JNDI 尋找和快取服務來取得服務的單點接觸。

  • 快取(Cache) - 快取儲存服務的引用,以便重複使用它們。

  • 客戶端(Client) - Client 是透過 ServiceLocator 呼叫服務的物件。

實作

我們將建立ServiceLocatorInitialContextCacheService 作為表示實體的各種物件。 Service1Service2 表示實體服務。

ServiceLocatorPatternDemo,我們的示範類別在這裡是作為一個客戶端,將使用 ServiceLocator 來示範服務定位器設計模式。

servicelocator_pattern_uml_diagram.jpg

步驟 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。

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