デザインパターンクイックガイド

デザインパターンは、経験豊富なオブジェクト指向ソフトウェア開発者が使用するベストプラクティスを表しています。デザインパターンは、ソフトウェア開発者がソフトウェア開発中に直面した一般的な問題の解決策です。これらのソリューションは、かなりの期間にわたって多くのソフトウェア開発者による試行錯誤によって得られました。

ギャングオブフォー(GOF)とは何ですか?

1994年に、4人の著者、エーリヒガンマ、リチャードヘルム、ラルフジョンソン、ジョンブリシディーズが、「 Design Patterns - Elements of Reusable Object-Oriented Software これは、ソフトウェア開発におけるデザインパターンの概念を開始しました。

これらの作者は総称して Gang of Four (GOF)。これらの著者によると、デザインパターンは主にオブジェクト指向設計の次の原則に基づいています。

  • 実装ではなくインターフェースへのプログラム

  • 継承よりもオブジェクトコンポジションを優先する

デザインパターンの使用法

デザインパターンには、ソフトウェア開発における2つの主な用途があります。

開発者向けの共通プラットフォーム

デザインパターンは標準的な用語を提供し、特定のシナリオに固有です。たとえば、シングルトンデザインパターンはシングルオブジェクトの使用を意味するため、シングルデザインパターンに精通しているすべての開発者はシングルオブジェクトを使用し、プログラムがシングルトンパターンに従っていることを互いに伝えることができます。

ベストプラクティス

デザインパターンは長い間進化してきており、ソフトウェア開発中に直面する特定の問題に対する最良の解決策を提供します。これらのパターンを学ぶことは、経験の浅い開発者が簡単かつ迅速な方法でソフトウェア設計を学ぶのに役立ちます。

デザインパターンの種類

デザインパターン参考書による Design Patterns - Elements of Reusable Object-Oriented Software、23のデザインパターンがあります。これらのパターンは、創造的、構造的、行動的パターンの3つのカテゴリに分類できます。また、デザインパターンの別のカテゴリであるJ2EEデザインパターンについても説明します。

SN パターンと説明
1 Creational Patterns
これらのデザインパターンは、new演算子を使用してオブジェクトを直接インスタンス化するのではなく、作成ロジックを非表示にしてオブジェクトを作成する方法を提供します。これにより、プログラムは、特定のユースケースで作成する必要のあるオブジェクトをより柔軟に決定できます。
2 Structural Patterns
これらのデザインパターンは、クラスとオブジェクトの構成に関係しています。継承の概念は、インターフェイスを構成し、オブジェクトを構成して新しい機能を取得する方法を定義するために使用されます。
3 Behavioral Patterns
これらのデザインパターンは、特にオブジェクト間の通信に関係しています。
4 J2EE Patterns
これらのデザインパターンは、特にプレゼンテーション層に関係しています。これらのパターンは、Sun JavaCenterによって識別されます。

ファクトリパターンは、Javaで最も使用されているデザインパターンの1つです。このタイプのデザインパターンは、オブジェクトを作成するための最良の方法の1つを提供するため、作成パターンに分類されます。

ファクトリパターンでは、作成ロジックをクライアントに公開せずにオブジェクトを作成し、共通のインターフェイスを使用して新しく作成されたオブジェクトを参照します。

実装

私たちは、作成しようとしているのShapeインタフェースと実装する具象クラスのShapeインタフェースを。次のステップとして、ファクトリクラスShapeFactoryが定義されています。

FactoryPatternDemo、デモクラスはShapeFactoryを使用してShapeオブジェクトを取得します。情報(CIRCLE / RECTANGLE / SQUARE)をShapeFactoryに渡して、必要なオブジェクトのタイプを取得します。

ステップ1

インターフェイスを作成します。

Shape.java

public interface Shape {
   void draw();
}

ステップ2

同じインターフェースを実装する具象クラスを作成します。

Rectangle.java

public class Rectangle implements Shape {

   @Override
   public void draw() {
      System.out.println("Inside Rectangle::draw() method.");
   }
}

Square.java

public class Square implements Shape {

   @Override
   public void draw() {
      System.out.println("Inside Square::draw() method.");
   }
}

Circle.java

public class Circle implements Shape {

   @Override
   public void draw() {
      System.out.println("Inside Circle::draw() method.");
   }
}

ステップ3

与えられた情報に基づいて具象クラスのオブジェクトを生成するファクトリを作成します。

ShapeFactory.java

public class ShapeFactory {
	
   //use getShape method to get object of type shape 
   public Shape getShape(String shapeType){
      if(shapeType == null){
         return null;
      }		
      if(shapeType.equalsIgnoreCase("CIRCLE")){
         return new Circle();
      } else if(shapeType.equalsIgnoreCase("RECTANGLE")){
         return new Rectangle();
      } else if(shapeType.equalsIgnoreCase("SQUARE")){
         return new Square();
      }
      return null;
   }
}

ステップ4

Factoryを使用して、型などの情報を渡すことにより、具象クラスのオブジェクトを取得します。

FactoryPatternDemo.java

public class FactoryPatternDemo {

   public static void main(String[] args) {
      ShapeFactory shapeFactory = new ShapeFactory();

      //get an object of Circle and call its draw method.
      Shape shape1 = shapeFactory.getShape("CIRCLE");

      //call draw method of Circle
      shape1.draw();

      //get an object of Rectangle and call its draw method.
      Shape shape2 = shapeFactory.getShape("RECTANGLE");

      //call draw method of Rectangle
      shape2.draw();

      //get an object of Square and call its draw method.
      Shape shape3 = shapeFactory.getShape("SQUARE");

      //call draw method of square
      shape3.draw();
   }
}

ステップ5

出力を確認します。

Inside Circle::draw() method.
Inside Rectangle::draw() method.
Inside Square::draw() method.

抽象ファクトリパターンは、他のファクトリを作成するスーパーファクトリを回避します。この工場は工場の工場とも呼ばれています。このタイプのデザインパターンは、オブジェクトを作成するための最良の方法の1つを提供するため、作成パターンに分類されます。

抽象ファクトリパターンでは、インターフェイスは、クラスを明示的に指定せずに、関連するオブジェクトのファクトリを作成する役割を果たします。生成された各ファクトリは、ファクトリパターンに従ってオブジェクトを提供できます。

実装

Shapeインターフェースとそれを実装する具象クラスを作成します。次のステップとして、抽象ファクトリクラスAbstractFactoryを作成します。AbstractFactoryを拡張するファクトリクラスShapeFactoryが定義されています。ファクトリクリエータ/ジェネレータクラスFactoryProducerが作成されます。

AbstractFactoryPatternDemo、デモクラスはFactoryProducerを使用してAbstractFactoryオブジェクトを取得します。情報(形状の場合はCIRCLE / RECTANGLE / SQUARE)をAbstractFactoryに渡して、必要なオブジェクトのタイプを取得します。

ステップ1

Shapesのインターフェースを作成します。

Shape.java

public interface Shape {
   void draw();
}

ステップ2

同じインターフェースを実装する具象クラスを作成します。

RoundedRectangle.java

public class RoundedRectangle implements Shape {
   @Override
   public void draw() {
      System.out.println("Inside RoundedRectangle::draw() method.");
   }
}

RoundedSquare.java

public class RoundedSquare implements Shape {
   @Override
   public void draw() {
      System.out.println("Inside RoundedSquare::draw() method.");
   }
}

Rectangle.java

public class Rectangle implements Shape {
   @Override
   public void draw() {
      System.out.println("Inside Rectangle::draw() method.");
   }
}

ステップ3

抽象クラスを作成して、ノーマルシェイプオブジェクトとラウンドシェイプオブジェクトのファクトリを取得します。

AbstractFactory.java

public abstract class AbstractFactory {
   abstract Shape getShape(String shapeType) ;
}

ステップ4

AbstractFactoryを拡張するファクトリクラスを作成して、指定された情報に基づいて具象クラスのオブジェクトを生成します。

ShapeFactory.java

public class ShapeFactory extends AbstractFactory {
   @Override
   public Shape getShape(String shapeType){    
      if(shapeType.equalsIgnoreCase("RECTANGLE")){
         return new Rectangle();         
      }else if(shapeType.equalsIgnoreCase("SQUARE")){
         return new Square();
      }	 
      return null;
   }
}

RoundedShapeFactory.java

public class RoundedShapeFactory extends AbstractFactory {
   @Override
   public Shape getShape(String shapeType){    
      if(shapeType.equalsIgnoreCase("RECTANGLE")){
         return new RoundedRectangle();         
      }else if(shapeType.equalsIgnoreCase("SQUARE")){
         return new RoundedSquare();
      }	 
      return null;
   }
}

ステップ5

シェイプなどの情報を渡してファクトリを取得するファクトリジェネレータ/プロデューサクラスを作成します

FactoryProducer.java

public class FactoryProducer {
   public static AbstractFactory getFactory(boolean rounded){   
      if(rounded){
         return new RoundedShapeFactory();         
      }else{
         return new ShapeFactory();
      }
   }
}

ステップ6

タイプなどの情報を渡して具象クラスのファクトリを取得するには、FactoryProducerを使用してAbstractFactoryを取得します。

AbstractFactoryPatternDemo.java

public class AbstractFactoryPatternDemo {
   public static void main(String[] args) {
      //get shape factory
      AbstractFactory shapeFactory = FactoryProducer.getFactory(false);
      //get an object of Shape Rectangle
      Shape shape1 = shapeFactory.getShape("RECTANGLE");
      //call draw method of Shape Rectangle
      shape1.draw();
      //get an object of Shape Square 
      Shape shape2 = shapeFactory.getShape("SQUARE");
      //call draw method of Shape Square
      shape2.draw();
      //get shape factory
      AbstractFactory shapeFactory1 = FactoryProducer.getFactory(true);
      //get an object of Shape Rectangle
      Shape shape3 = shapeFactory1.getShape("RECTANGLE");
      //call draw method of Shape Rectangle
      shape3.draw();
      //get an object of Shape Square 
      Shape shape4 = shapeFactory1.getShape("SQUARE");
      //call draw method of Shape Square
      shape4.draw();
      
   }
}

ステップ7

出力を確認します。

Inside Rectangle::draw() method.
Inside Square::draw() method.
Inside RoundedRectangle::draw() method.
Inside RoundedSquare::draw() method.

シングルトンパターンは、Javaで最も単純なデザインパターンの1つです。このタイプのデザインパターンは、オブジェクトを作成するための最良の方法の1つを提供するため、作成パターンに分類されます。

このパターンには、単一のオブジェクトのみが作成されるようにしながら、独自のオブジェクトを作成する責任がある単一のクラスが含まれます。このクラスは、クラスのオブジェクトをインスタンス化する必要なしに直接アクセスできる唯一のオブジェクトにアクセスする方法を提供します。

実装

SingleObjectクラスを作成します。SingleObjectクラスには、そのコンストラクターがプライベートであり、それ自体の静的インスタンスがあります。

SingleObjectクラスは、静的インスタンスを外部に取得するための静的メソッドを提供します。SingletonPatternDemo、デモクラスはSingleObjectクラスを使用してSingleObjectオブジェクトを取得します。

ステップ1

シングルトンクラスを作成します。

SingleObject.java

public class SingleObject {

   //create an object of SingleObject
   private static SingleObject instance = new SingleObject();

   //make the constructor private so that this class cannot be
   //instantiated
   private SingleObject(){}

   //Get the only object available
   public static SingleObject getInstance(){
      return instance;
   }

   public void showMessage(){
      System.out.println("Hello World!");
   }
}

ステップ2

シングルトンクラスから唯一のオブジェクトを取得します。

SingletonPatternDemo.java

public class SingletonPatternDemo {
   public static void main(String[] args) {

      //illegal construct
      //Compile Time Error: The constructor SingleObject() is not visible
      //SingleObject object = new SingleObject();

      //Get the only object available
      SingleObject object = SingleObject.getInstance();

      //show the message
      object.showMessage();
   }
}

ステップ3

出力を確認します。

Hello World!

Builderパターンは、単純なオブジェクトを使用し、段階的なアプローチを使用して複雑なオブジェクトを構築します。このタイプのデザインパターンは、オブジェクトを作成するための最良の方法の1つを提供するため、作成パターンに分類されます。

Builderクラスは、最終的なオブジェクトを段階的に構築します。このビルダーは他のオブジェクトから独立しています。

実装

典型的な食事がハンバーガーと冷たい飲み物であるファーストフードレストランのビジネスケースを検討しました。バーガーはベジーバーガーまたはチキンバーガーのいずれかで、ラッパーで梱包されます。冷たい飲み物はコーラまたはペプシのいずれかであり、ボトルに詰められます。

私たちは、作成しようとしている項目などハンバーガーや冷たい飲み物や実装する具象クラスとして食品表すインターフェース項目インターフェースおよび包装食品および実装する具象クラスの実装表すインタフェースパッキングラッパーと寒さの中に詰められるバーガーなどのインタフェースを飲み物はボトルとして梱包されます。

私たちは、その後、作成食事持つクラスをArrayListの項目MealBuilder、異なる種類の構築に食事組み合わせることで、オブジェクトを項目BuilderPatternDemo、私たちのデモクラスが使用されますMealBuilderを構築するために食事を

ステップ1

食品とパッキングを表すインターフェースItemを作成します。

Item.java

public interface Item {
   public String name();
   public Packing packing();
   public float price();	
}

Packing.java

public interface Packing {
   public String pack();
}

ステップ2

Packingインターフェースを実装するconcreateクラスを作成します。

Wrapper.java

public class Wrapper implements Packing {

   @Override
   public String pack() {
      return "Wrapper";
   }
}

Bottle.java

public class Bottle implements Packing {

   @Override
   public String pack() {
      return "Bottle";
   }
}

ステップ3

デフォルトの機能を提供するアイテムインターフェイスを実装する抽象クラスを作成します。

Burger.java

public abstract class Burger implements Item {

   @Override
   public Packing packing() {
      return new Wrapper();
   }

   @Override
   public abstract float price();
}

ColdDrink.java

public abstract class ColdDrink implements Item {

	@Override
	public Packing packing() {
       return new Bottle();
	}

	@Override
	public abstract float price();
}

ステップ4

BurgerクラスとColdDrinkクラスを拡張する具象クラスを作成する

VegBurger.java

public class VegBurger extends Burger {

   @Override
   public float price() {
      return 25.0f;
   }

   @Override
   public String name() {
      return "Veg Burger";
   }
}

ChickenBurger.java

public class ChickenBurger extends Burger {

   @Override
   public float price() {
      return 50.5f;
   }

   @Override
   public String name() {
      return "Chicken Burger";
   }
}

Coke.java

public class Coke extends ColdDrink {

   @Override
   public float price() {
      return 30.0f;
   }

   @Override
   public String name() {
      return "Coke";
   }
}

Pepsi.java

public class Pepsi extends ColdDrink {

   @Override
   public float price() {
      return 35.0f;
   }

   @Override
   public String name() {
      return "Pepsi";
   }
}

ステップ5

上で定義したItemオブジェクトを持つMealクラスを作成します。

Meal.java

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

public class Meal {
   private List<Item> items = new ArrayList<Item>();	

   public void addItem(Item item){
      items.add(item);
   }

   public float getCost(){
      float cost = 0.0f;
      for (Item item : items) {
         cost += item.price();
      }		
      return cost;
   }

   public void showItems(){
      for (Item item : items) {
         System.out.print("Item : "+item.name());
         System.out.print(", Packing : "+item.packing().pack());
         System.out.println(", Price : "+item.price());
      }		
   }	
}

ステップ6

MealBuilderクラスを作成します。これは、Mealオブジェクトの作成を担当する実際のビルダークラスです。

MealBuilder.java

public class MealBuilder {

   public Meal prepareVegMeal (){
      Meal meal = new Meal();
      meal.addItem(new VegBurger());
      meal.addItem(new Coke());
      return meal;
   }   

   public Meal prepareNonVegMeal (){
      Meal meal = new Meal();
      meal.addItem(new ChickenBurger());
      meal.addItem(new Pepsi());
      return meal;
   }
}

ステップ7

BuiderPatternDemoは、MealBuiderを使用してビルダーパターンを示します。

BuilderPatternDemo.java

public class BuilderPatternDemo {
   public static void main(String[] args) {
      MealBuilder mealBuilder = new MealBuilder();

      Meal vegMeal = mealBuilder.prepareVegMeal();
      System.out.println("Veg Meal");
      vegMeal.showItems();
      System.out.println("Total Cost: " +vegMeal.getCost());

      Meal nonVegMeal = mealBuilder.prepareNonVegMeal();
      System.out.println("\n\nNon-Veg Meal");
      nonVegMeal.showItems();
      System.out.println("Total Cost: " +nonVegMeal.getCost());
   }
}

ステップ8

出力を確認します。

Veg Meal
Item : Veg Burger, Packing : Wrapper, Price : 25.0
Item : Coke, Packing : Bottle, Price : 30.0
Total Cost: 55.0

Non-Veg Meal
Item : Chicken Burger, Packing : Wrapper, Price : 50.5
Item : Pepsi, Packing : Bottle, Price : 35.0
Total Cost: 85.5

プロトタイプパターンとは、パフォーマンスを念頭に置いて重複オブジェクトを作成することです。このタイプのデザインパターンは、オブジェクトを作成するための最良の方法の1つを提供するため、作成パターンに分類されます。

このパターンには、現在のオブジェクトのクローンを作成するように指示するプロトタイプインターフェイスの実装が含まれます。このパターンは、オブジェクトを直接作成するのにコストがかかる場合に使用されます。たとえば、オブジェクトは、コストのかかるデータベース操作の後に作成されます。オブジェクトをキャッシュし、次のリクエストでそのクローンを返し、必要に応じてデータベースを更新することで、データベースの呼び出しを減らすことができます。

実装

抽象クラスShapeと、Shapeクラスを拡張する具象クラスを作成します。クラスShapeCacheは、シェイプオブジェクトをハッシュテーブルに格納し、要求されたときにそれらのクローンを返す次のステップとして定義されます。

PrototypPatternDemo、デモクラスはShapeCacheクラスを使用してShapeオブジェクトを取得します。

ステップ1

Clonableインターフェースを実装する抽象クラスを作成します。

Shape.java

public abstract class Shape implements Cloneable {
   
   private String id;
   protected String type;
   
   abstract void draw();
   
   public String getType(){
      return type;
   }
   
   public String getId() {
      return id;
   }
   
   public void setId(String id) {
      this.id = id;
   }
   
   public Object clone() {
      Object clone = null;
      try {
         clone = super.clone();
      } catch (CloneNotSupportedException e) {
         e.printStackTrace();
      }
      return clone;
   }
}

ステップ2

上記のクラスを拡張する具象クラスを作成します。

Rectangle.java

public class Rectangle extends Shape {

   public Rectangle(){
     type = "Rectangle";
   }

   @Override
   public void draw() {
      System.out.println("Inside Rectangle::draw() method.");
   }
}

Square.java

public class Square extends Shape {

   public Square(){
     type = "Square";
   }

   @Override
   public void draw() {
      System.out.println("Inside Square::draw() method.");
   }
}

Circle.java

public class Circle extends Shape {

   public Circle(){
     type = "Circle";
   }

   @Override
   public void draw() {
      System.out.println("Inside Circle::draw() method.");
   }
}

ステップ3

クラスを作成して、データベースからconcreateクラスを取得し、それらをハッシュテーブルに格納します。

ShapeCache.java

import java.util.Hashtable;

public class ShapeCache {
	
   private static Hashtable<String, Shape> shapeMap 
      = new Hashtable<String, Shape>();

   public static Shape getShape(String shapeId) {
      Shape cachedShape = shapeMap.get(shapeId);
      return (Shape) cachedShape.clone();
   }

   // for each shape run database query and create shape
   // shapeMap.put(shapeKey, shape);
   // for example, we are adding three shapes
   public static void loadCache() {
      Circle circle = new Circle();
      circle.setId("1");
      shapeMap.put(circle.getId(),circle);

      Square square = new Square();
      square.setId("2");
      shapeMap.put(square.getId(),square);

      Rectangle rectangle = new Rectangle();
      rectangle.setId("3");
      shapeMap.put(rectangle.getId(),rectangle);
   }
}

ステップ4

PrototypePatternDemoは、ShapeCacheクラスを使用して、ハッシュテーブルに格納されているシェイプのクローンを取得します。

PrototypePatternDemo.java

public class PrototypePatternDemo {
   public static void main(String[] args) {
      ShapeCache.loadCache();

      Shape clonedShape = (Shape) ShapeCache.getShape("1");
      System.out.println("Shape : " + clonedShape.getType());		

      Shape clonedShape2 = (Shape) ShapeCache.getShape("2");
      System.out.println("Shape : " + clonedShape2.getType());		

      Shape clonedShape3 = (Shape) ShapeCache.getShape("3");
      System.out.println("Shape : " + clonedShape3.getType());		
   }
}

ステップ5

出力を確認します。

Shape : Circle
Shape : Square
Shape : Rectangle

アダプタパターンは、互換性のない2つのインターフェイス間のブリッジとして機能します。このタイプのデザインパターンは、2つの独立したインターフェイスの機能を組み合わせているため、構造パターンに分類されます。

このパターンには、独立したインターフェイスまたは互換性のないインターフェイスの機能を結合する責任がある単一のクラスが含まれます。実際の例としては、メモリカードとラップトップの間のアダプタとして機能するカードリーダーの場合があります。メモリカードをカードリーダーに接続し、カードリーダーをラップトップに接続して、ラップトップからメモリカードを読み取れるようにします。

オーディオプレーヤーデバイスがmp3ファイルのみを再生でき、vlcおよびmp4ファイルを再生できる高度なオーディオプレーヤーを使用したいという次の例で、アダプターパターンの使用法を示しています。

実装

私たちは、インターフェイスましMediaPlayerのインターフェースと具象クラスAudioPlayer実装MediaPlayerののインターフェイスを。AudioPlayerは、デフォルトでmp3形式のオーディオファイルを再生できます。

別のインターフェイスAdvancedMediaPlayerと、AdvancedMediaPlayerインターフェイスを実装する具象クラスがあります。これらのクラスはvlcおよびmp4形式のファイルを再生できます。

AudioPlayerで他のフォーマットも再生できるようにしたいと考えています。これを実現するために、MediaPlayerインターフェイスを実装し、AdvancedMediaPlayerオブジェクトを使用して必要な形式を再生するアダプタクラスMediaAdapterを作成しました。

AudioPlayerは、アダプタクラスMediaAdapterを使用して、目的の形式を再生できる実際のクラスを知らなくても、目的のオーディオタイプを渡します。AdapterPatternDemo、デモクラスはAudioPlayerクラスを使用してさまざまなフォーマットを再生します。

ステップ1

MediaPlayerおよびAdvancedMediaPlayerのインターフェイスを作成します。

MediaPlayer.java

public interface MediaPlayer {
   public void play(String audioType, String fileName);
}

AdvancedMediaPlayer.java

public interface AdvancedMediaPlayer {	
   public void playVlc(String fileName);
   public void playMp4(String fileName);
}

ステップ2

AdvancedMediaPlayerインターフェイスを実装する具象クラスを作成します。

VlcPlayer.java

public class VlcPlayer implements AdvancedMediaPlayer{
   @Override
   public void playVlc(String fileName) {
      System.out.println("Playing vlc file. Name: "+ fileName);		
   }

   @Override
   public void playMp4(String fileName) {
      //do nothing
   }
}

Mp4Player.java

public class Mp4Player implements AdvancedMediaPlayer{

   @Override
   public void playVlc(String fileName) {
      //do nothing
   }

   @Override
   public void playMp4(String fileName) {
      System.out.println("Playing mp4 file. Name: "+ fileName);		
   }
}

ステップ3

MediaPlayerインターフェイスを実装するアダプタクラスを作成します。

MediaAdapter.java

public class MediaAdapter implements MediaPlayer {

   AdvancedMediaPlayer advancedMusicPlayer;

   public MediaAdapter(String audioType){
      if(audioType.equalsIgnoreCase("vlc") ){
         advancedMusicPlayer = new VlcPlayer();			
      } else if (audioType.equalsIgnoreCase("mp4")){
         advancedMusicPlayer = new Mp4Player();
      }	
   }

   @Override
   public void play(String audioType, String fileName) {
      if(audioType.equalsIgnoreCase("vlc")){
         advancedMusicPlayer.playVlc(fileName);
      }else if(audioType.equalsIgnoreCase("mp4")){
         advancedMusicPlayer.playMp4(fileName);
      }
   }
}

ステップ4

MediaPlayerインターフェイスを実装する具象クラスを作成します。

AudioPlayer.java

public class AudioPlayer implements MediaPlayer {
   MediaAdapter mediaAdapter; 

   @Override
   public void play(String audioType, String fileName) {		

      //inbuilt support to play mp3 music files
      if(audioType.equalsIgnoreCase("mp3")){
         System.out.println("Playing mp3 file. Name: "+ fileName);			
      } 
      //mediaAdapter is providing support to play other file formats
      else if(audioType.equalsIgnoreCase("vlc") 
         || audioType.equalsIgnoreCase("mp4")){
         mediaAdapter = new MediaAdapter(audioType);
         mediaAdapter.play(audioType, fileName);
      }
      else{
         System.out.println("Invalid media. "+
            audioType + " format not supported");
      }
   }   
}

ステップ5

AudioPlayerを使用して、さまざまな種類のオーディオ形式を再生します。

AdapterPatternDemo.java

public class AdapterPatternDemo {
   public static void main(String[] args) {
      AudioPlayer audioPlayer = new AudioPlayer();

      audioPlayer.play("mp3", "beyond the horizon.mp3");
      audioPlayer.play("mp4", "alone.mp4");
      audioPlayer.play("vlc", "far far away.vlc");
      audioPlayer.play("avi", "mind me.avi");
   }
}

ステップ6

出力を確認します。

Playing mp3 file. Name: beyond the horizon.mp3
Playing mp4 file. Name: alone.mp4
Playing vlc file. Name: far far away.vlc
Invalid media. avi format not supported

Bridgeは、抽象化をその実装から切り離して、2つを独立して変化させる必要がある場合に使用されます。このタイプのデザインパターンは、実装クラスと抽象クラスの間にブリッジ構造を提供することでそれらを分離するため、構造パターンに分類されます。

このパターンには、具象クラスの機能をインターフェース実装クラスから独立させるブリッジとして機能するインターフェースが含まれます。両方のタイプのクラスは、互いに影響を与えることなく構造的に変更できます。

同じ抽象クラスメソッドを使用して、異なるブリッジ実装クラスを使用して円を異なる色で描画できる次の例を使用して、ブリッジパターンの使用法を示しています。

実装

ブリッジ実装者として機能するインターフェイスDrawAPIインターフェイスと、DrawAPIインターフェイスを実装する具体的なクラスRedCircleGreenCircleあります。Shapeは抽象クラスであり、DrawAPIのオブジェクトを使用します。BridgePatternDemo、デモクラスはShapeクラスを使用して異なる色の円を描画します。

ステップ1

ブリッジ実装インターフェイスを作成します。

DrawAPI.java

public interface DrawAPI {
   public void drawCircle(int radius, int x, int y);
}

ステップ2

DrawAPIインターフェイスを実装する具象橋実装クラスを作成します。

RedCircle.java

public class RedCircle implements DrawAPI {
   @Override
   public void drawCircle(int radius, int x, int y) {
      System.out.println("Drawing Circle[ color: red, radius: "
         + radius +", x: " +x+", "+ y +"]");
   }
}

GreenCircle.java

public class GreenCircle implements DrawAPI {
   @Override
   public void drawCircle(int radius, int x, int y) {
      System.out.println("Drawing Circle[ color: green, radius: "
         + radius +", x: " +x+", "+ y +"]");
   }
}

ステップ3

DrawAPIインターフェイスを使用して抽象クラスShapeを作成します。

Shape.java

public abstract class Shape {
   protected DrawAPI drawAPI;
   protected Shape(DrawAPI drawAPI){
      this.drawAPI = drawAPI;
   }
   public abstract void draw();	
}

ステップ4

Shapeインターフェースを実装する具象クラスを作成します。

Circle.java

public class Circle extends Shape {
   private int x, y, radius;

   public Circle(int x, int y, int radius, DrawAPI drawAPI) {
      super(drawAPI);
      this.x = x;  
      this.y = y;  
      this.radius = radius;
   }

   public void draw() {
      drawAPI.drawCircle(radius,x,y);
   }
}

ステップ5

ShapeクラスとDrawAPIクラスを使用して、さまざまな色の円を描画します。

BridgePatternDemo.java

public class BridgePatternDemo {
   public static void main(String[] args) {
      Shape redCircle = new Circle(100,100, 10, new RedCircle());
      Shape greenCircle = new Circle(100,100, 10, new GreenCircle());

      redCircle.draw();
      greenCircle.draw();
   }
}

ステップ6

出力を確認します。

Drawing Circle[ color: red, radius: 10, x: 100, 100]
Drawing Circle[  color: green, radius: 10, x: 100, 100]

フィルタパターンまたは基準パターンは、開発者がさまざまな基準を使用してオブジェクトのセットをフィルタリングし、論理演算を通じて分離された方法でオブジェクトをチェーンできるようにするデザインパターンです。このタイプのデザインパターンは、複数の基準を組み合わせて単一の基準を取得するため、構造パターンに分類されます。

実装

Personオブジェクト、Criteriaインターフェイス、およびこのインターフェイスを実装してPersonオブジェクトのリストをフィルタリングする具象クラスを作成します。CriteriaPatternDemo、私たちのデモクラスの用途の基準のフィルタリストへのオブジェクト人は、様々な基準やその組み合わせに基づいてオブジェクト。

ステップ1

基準を適用するクラスを作成します。

Person.java

public class Person {
	
   private String name;
   private String gender;
   private String maritalStatus;

   public Person(String name,String gender,String maritalStatus){
      this.name = name;
      this.gender = gender;
      this.maritalStatus = maritalStatus;		
   }

   public String getName() {
      return name;
   }
   public String getGender() {
      return gender;
   }
   public String getMaritalStatus() {
      return maritalStatus;
   }	
}

ステップ2

Criteriaのインターフェースを作成します。

Criteria.java

import java.util.List;

public interface Criteria {
   public List<Person> meetCriteria(List<Person> persons);
}

ステップ3

Criteriaインターフェースを実装する具象クラスを作成します。

CriteriaMale.java

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

public class CriteriaMale implements Criteria {

   @Override
   public List<Person> meetCriteria(List<Person> persons) {
      List<Person> malePersons = new ArrayList<Person>(); 
      for (Person person : persons) {
         if(person.getGender().equalsIgnoreCase("MALE")){
            malePersons.add(person);
         }
      }
      return malePersons;
   }
}

CriteriaFemale.java

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

public class CriteriaFemale implements Criteria {

   @Override
   public List<Person> meetCriteria(List<Person> persons) {
      List<Person> femalePersons = new ArrayList<Person>(); 
      for (Person person : persons) {
         if(person.getGender().equalsIgnoreCase("FEMALE")){
            femalePersons.add(person);
         }
      }
      return femalePersons;
   }
}

CriteriaSingle.java

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

public class CriteriaSingle implements Criteria {

   @Override
   public List<Person> meetCriteria(List<Person> persons) {
      List<Person> singlePersons = new ArrayList<Person>(); 
      for (Person person : persons) {
         if(person.getMaritalStatus().equalsIgnoreCase("SINGLE")){
            singlePersons.add(person);
         }
      }
      return singlePersons;
   }
}

AndCriteria.java

import java.util.List;

public class AndCriteria implements Criteria {

   private Criteria criteria;
   private Criteria otherCriteria;

   public AndCriteria(Criteria criteria, Criteria otherCriteria) {
      this.criteria = criteria;
      this.otherCriteria = otherCriteria; 
   }

   @Override
   public List<Person> meetCriteria(List<Person> persons) {
      List<Person> firstCriteriaPersons = criteria.meetCriteria(persons);		
      return otherCriteria.meetCriteria(firstCriteriaPersons);
   }
}

OrCriteria.java

import java.util.List;

public class AndCriteria implements Criteria {

   private Criteria criteria;
   private Criteria otherCriteria;

   public AndCriteria(Criteria criteria, Criteria otherCriteria) {
      this.criteria = criteria;
      this.otherCriteria = otherCriteria; 
   }

   @Override
   public List<Person> meetCriteria(List<Person> persons) {
      List<Person> firstCriteriaItems = criteria.meetCriteria(persons);
      List<Person> otherCriteriaItems = otherCriteria.meetCriteria(persons);

      for (Person person : otherCriteriaItems) {
         if(!firstCriteriaItems.contains(person)){
	        firstCriteriaItems.add(person);
         }
      }	
      return firstCriteriaItems;
   }
}

ステップ4

さまざまな基準とその組み合わせを使用して、人を除外します。

CriteriaPatternDemo.java

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

public class CriteriaPatternDemo {
   public static void main(String[] args) {
      List<Person> persons = new ArrayList<Person>();

      persons.add(new Person("Robert","Male", "Single"));
      persons.add(new Person("John","Male", "Married"));
      persons.add(new Person("Laura","Female", "Married"));
      persons.add(new Person("Diana","Female", "Single"));
      persons.add(new Person("Mike","Male", "Single"));
      persons.add(new Person("Bobby","Male", "Single"));

      Criteria male = new CriteriaMale();
      Criteria female = new CriteriaFemale();
      Criteria single = new CriteriaSingle();
      Criteria singleMale = new AndCriteria(single, male);
      Criteria singleOrFemale = new OrCriteria(single, female);

      System.out.println("Males: ");
      printPersons(male.meetCriteria(persons));

      System.out.println("\nFemales: ");
      printPersons(female.meetCriteria(persons));

      System.out.println("\nSingle Males: ");
      printPersons(singleMale.meetCriteria(persons));

      System.out.println("\nSingle Or Females: ");
      printPersons(singleOrFemale.meetCriteria(persons));
   }

   public static void printPersons(List<Person> persons){
      for (Person person : persons) {
         System.out.println("Person : [ Name : " + person.getName() 
            +", Gender : " + person.getGender() 
            +", Marital Status : " + person.getMaritalStatus()
            +" ]");
      }
   }      
}

ステップ5

出力を確認します。

Males: 
Person : [ Name : Robert, Gender : Male, Marital Status : Single ]
Person : [ Name : John, Gender : Male, Marital Status : Married ]
Person : [ Name : Mike, Gender : Male, Marital Status : Single ]
Person : [ Name : Bobby, Gender : Male, Marital Status : Single ]

Females: 
Person : [ Name : Laura, Gender : Female, Marital Status : Married ]
Person : [ Name : Diana, Gender : Female, Marital Status : Single ]

Single Males: 
Person : [ Name : Robert, Gender : Male, Marital Status : Single ]
Person : [ Name : Mike, Gender : Male, Marital Status : Single ]
Person : [ Name : Bobby, Gender : Male, Marital Status : Single ]

Single Or Females: 
Person : [ Name : Robert, Gender : Male, Marital Status : Single ]
Person : [ Name : Diana, Gender : Female, Marital Status : Single ]
Person : [ Name : Mike, Gender : Male, Marital Status : Single ]
Person : [ Name : Bobby, Gender : Male, Marital Status : Single ]
Person : [ Name : Laura, Gender : Female, Marital Status : Married ]

複合パターンは、オブジェクトのグループを単一のオブジェクトと同じように扱う必要がある場合に使用されます。複合パターンは、階層全体だけでなく一部を表すために、ツリー構造の観点からオブジェクトを構成します。このタイプのデザインパターンは、オブジェクトのグループのツリー構造を作成するため、構造パターンに分類されます。

このパターンは、独自のオブジェクトのグループを含むクラスを作成します。このクラスは、同じオブジェクトのグループを変更する方法を提供します。

組織の従業員階層を示す次の例を使用して、複合パターンの使用を示しています。

実装

複合パターンアクタークラスとして機能するクラスEmployeeがあります。CompositePatternDemo、デモクラスは、Employeeクラスを使用して部門レベルの階層を追加し、すべての従業員を印刷します。

ステップ1

Employeeオブジェクトのリストを持つEmployeeクラスを作成します。

Employee.java

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

public class Employee {
   private String name;
   private String dept;
   private int salary;
   private List<Employee> subordinates;

   // constructor
   public Employee(String name,String dept, int sal) {
      this.name = name;
      this.dept = dept;
      this.salary = sal;
      subordinates = new ArrayList<Employee>();
   }

   public void add(Employee e) {
      subordinates.add(e);
   }

   public void remove(Employee e) {
      subordinates.remove(e);
   }

   public List<Employee> getSubordinates(){
     return subordinates;
   }

   public String toString(){
      return ("Employee :[ Name : "+ name 
      +", dept : "+ dept + ", salary :"
      + salary+" ]");
   }   
}

ステップ2

Employeeクラスを使用して、従業員階層を作成および印刷します。

CompositePatternDemo.java

public class CompositePatternDemo {
   public static void main(String[] args) {
      Employee CEO = new Employee("John","CEO", 30000);

      Employee headSales = new Employee("Robert","Head Sales", 20000);

      Employee headMarketing = new Employee("Michel","Head Marketing", 20000);

      Employee clerk1 = new Employee("Laura","Marketing", 10000);
      Employee clerk2 = new Employee("Bob","Marketing", 10000);

      Employee salesExecutive1 = new Employee("Richard","Sales", 10000);
      Employee salesExecutive2 = new Employee("Rob","Sales", 10000);

      CEO.add(headSales);
      CEO.add(headMarketing);

      headSales.add(salesExecutive1);
      headSales.add(salesExecutive2);

      headMarketing.add(clerk1);
      headMarketing.add(clerk2);

      //print all employees of the organization
      System.out.println(CEO); 
      for (Employee headEmployee : CEO.getSubordinates()) {
         System.out.println(headEmployee);
         for (Employee employee : headEmployee.getSubordinates()) {
            System.out.println(employee);
         }
      }		
   }
}

ステップ3

出力を確認します。

Employee :[ Name : John, dept : CEO, salary :30000 ]
Employee :[ Name : Robert, dept : Head Sales, salary :20000 ]
Employee :[ Name : Richard, dept : Sales, salary :10000 ]
Employee :[ Name : Rob, dept : Sales, salary :10000 ]
Employee :[ Name : Michel, dept : Head Marketing, salary :20000 ]
Employee :[ Name : Laura, dept : Marketing, salary :10000 ]
Employee :[ Name : Bob, dept : Marketing, salary :10000 ]

デコレータパターンを使用すると、構造を変更せずに既存のオブジェクトに新しい機能を追加できます。このタイプのデザインパターンは、このパターンが既存のクラスのラッパーとして機能するため、構造パターンに分類されます。

このパターンは、元のクラスをラップし、クラスメソッドのシグネチャをそのまま維持する追加機能を提供するデコレータクラスを作成します。

次の例では、デコレータパターンの使用方法を示しています。この例では、形状クラスを変更せずに、ある色で形状を装飾します。

実装

私たちは、作成しようとしているのShapeインタフェースと実装する具象クラスのShapeインタフェースを。次に、Shapeインターフェイスを実装し、インスタンス変数としてShapeオブジェクトを持つ抽象デコレータクラスShapeDecoratorを作成します。

RedShapeDecoratorは、ShapeDecoratorを実装する具象クラスです

DecoratorPatternDemo、デモクラスはRedShapeDecoratorを使用してShapeオブジェクトを装飾します。

ステップ1

インターフェイスを作成します。

Shape.java

public interface Shape {
   void draw();
}

ステップ2

同じインターフェースを実装する具象クラスを作成します。

Rectangle.java

public class Rectangle implements Shape {

   @Override
   public void draw() {
      System.out.println("Shape: Rectangle");
   }
}

Circle.java

public class Circle implements Shape {

   @Override
   public void draw() {
      System.out.println("Shape: Circle");
   }
}

ステップ3

Shapeインターフェイスを実装する抽象デコレータクラスを作成します。

ShapeDecorator.java

public abstract class ShapeDecorator implements Shape {
   protected Shape decoratedShape;

   public ShapeDecorator(Shape decoratedShape){
      this.decoratedShape = decoratedShape;
   }

   public void draw(){
      decoratedShape.draw();
   }	
}

ステップ4

ShapeDecoratorクラスを拡張する具象デコレータクラスを作成します。

RedShapeDecorator.java

public class RedShapeDecorator extends ShapeDecorator {

   public RedShapeDecorator(Shape decoratedShape) {
      super(decoratedShape);		
   }

   @Override
   public void draw() {
      decoratedShape.draw();	       
      setRedBorder(decoratedShape);
   }

   private void setRedBorder(Shape decoratedShape){
      System.out.println("Border Color: Red");
   }
}

ステップ5

RedShapeDecoratorを使用して、Shapeオブジェクトを装飾します。

DecoratorPatternDemo.java

public class DecoratorPatternDemo {
   public static void main(String[] args) {

      Shape circle = new Circle();

      Shape redCircle = new RedShapeDecorator(new Circle());

      Shape redRectangle = new RedShapeDecorator(new Rectangle());
      System.out.println("Circle with normal border");
      circle.draw();

      System.out.println("\nCircle of red border");
      redCircle.draw();

      System.out.println("\nRectangle of red border");
      redRectangle.draw();
   }
}

ステップ6

出力を確認します。

Circle with normal border
Shape: Circle

Circle of red border
Shape: Circle
Border Color: Red

Rectangle of red border
Shape: Rectangle
Border Color: Red

ファサードパターンは、システムの複雑さを隠し、クライアントがシステムにアクセスできるためのインターフェイスをクライアントに提供します。このタイプのデザインパターンは、複雑さを隠すために既存のシステムへのインターフェイスを追加するため、構造パターンに分類されます。

このパターンには、クライアントが必要とする単純化されたメソッドを提供し、既存のシステムクラスメソッドへの呼び出しを委任する単一のクラスが含まれます。

実装

私たちは、作成しようとしているのShapeインタフェースと実装する具象クラスのShapeインタフェースを。次のステップとして、ファサードクラスShapeMakerが定義されています。

ShapeMakerクラスは、具象クラスを使用して、ユーザー呼び出しをこれらのクラスに委任します。FacadePatternDemo、デモクラスはShapeMakerクラスを使用して結果を表示します。

ステップ1

インターフェイスを作成します。

Shape.java

public interface Shape {
   void draw();
}

ステップ2

同じインターフェースを実装する具象クラスを作成します。

Rectangle.java

public class Rectangle implements Shape {

   @Override
   public void draw() {
      System.out.println("Rectangle::draw()");
   }
}

Square.java

public class Square implements Shape {

   @Override
   public void draw() {
      System.out.println("Square::draw()");
   }
}

Circle.java

public class Circle implements Shape {

   @Override
   public void draw() {
      System.out.println("Circle::draw()");
   }
}

ステップ3

ファサードクラスを作成します。

ShapeMaker.java

public class ShapeMaker {
   private Shape circle;
   private Shape rectangle;
   private Shape square;

   public ShapeMaker() {
      circle = new Circle();
      rectangle = new Rectangle();
      square = new Square();
   }

   public void drawCircle(){
      circle.draw();
   }
   public void drawRectangle(){
      rectangle.draw();
   }
   public void drawSquare(){
      square.draw();
   }
}

ステップ4

ファサードを使用して、さまざまなタイプの形状を描画します。

FacadePatternDemo.java

public class FacadePatternDemo {
   public static void main(String[] args) {
      ShapeMaker shapeMaker = new ShapeMaker();

      shapeMaker.drawCircle();
      shapeMaker.drawRectangle();
      shapeMaker.drawSquare();		
   }
}

ステップ5

出力を確認します。

Circle::draw()
Rectangle::draw()
Square::draw()

Flyweightパターンは主に、作成されるオブジェクトの数を減らし、メモリフットプリントを減らし、パフォーマンスを向上させるために使用されます。このタイプのデザインパターンは、オブジェクト数を減らしてアプリケーションに必要なオブジェクト構造を改善する方法を提供するため、構造パターンに分類されます。

Flyweightパターンは、既存の同様の種類のオブジェクトを保存して再利用しようとし、一致するオブジェクトが見つからない場合は新しいオブジェクトを作成します。さまざまな場所に20個の円を描くことでこのパターンを示しますが、作成するオブジェクトは5つだけです。5色しか使用できないため、colorプロパティを使用して既存のCircleオブジェクトをチェックします。

実装

私たちは、作成しようとしているのShapeインターフェースと具象クラスサークルが実装シェイプインタフェースを。次のステップとして、ファクトリクラスShapeFactoryが定義されています。

ShapeFactoryには、Circleオブジェクトの色としてキーを持つCircleのHashMapがあります。ShapeFactoryに特定の色の円を作成するように要求されたときはいつでも。ShapeFactoryは、HashMapCircleオブジェクトをチェックし、Circleのオブジェクトが見つかった場合はそのオブジェクトが返されます。それ以外の場合は、新しいオブジェクトが作成され、将来使用するためにハッシュマップに保存されてクライアントに返されます。

FlyWeightPatternDemo、デモクラスはShapeFactoryを使用してShapeオブジェクトを取得します。情報(赤/緑/青/黒/白)をShapeFactoryに渡して、必要な色の円を取得します。

ステップ1

インターフェイスを作成します。

Shape.java

public interface Shape {
   void draw();
}

ステップ2

同じインターフェースを実装する具象クラスを作成します。

Circle.java

public class Circle implements Shape {
   private String color;
   private int x;
   private int y;
   private int radius;

   public Circle(String color){
      this.color = color;		
   }

   public void setX(int x) {
      this.x = x;
   }

   public void setY(int y) {
      this.y = y;
   }

   public void setRadius(int radius) {
      this.radius = radius;
   }

   @Override
   public void draw() {
      System.out.println("Circle: Draw() [Color : " + color 
         +", x : " + x +", y :" + y +", radius :" + radius);
   }
}

ステップ3

与えられた情報に基づいて具象クラスのオブジェクトを生成するファクトリを作成します。

ShapeFactory.java

import java.util.HashMap;

public class ShapeFactory {

   // Uncomment the compiler directive line and
   // javac *.java will compile properly.
   // @SuppressWarnings("unchecked")
   private static final HashMap circleMap = new HashMap();

   public static Shape getCircle(String color) {
      Circle circle = (Circle)circleMap.get(color);

      if(circle == null) {
         circle = new Circle(color);
         circleMap.put(color, circle);
         System.out.println("Creating circle of color : " + color);
      }
      return circle;
   }
}

ステップ4

Factoryを使用して、色などの情報を渡すことにより、具象クラスのオブジェクトを取得します。

FlyweightPatternDemo.java

public class FlyweightPatternDemo {
   private static final String colors[] = 
      { "Red", "Green", "Blue", "White", "Black" };
   public static void main(String[] args) {

      for(int i=0; i < 20; ++i) {
         Circle circle = 
            (Circle)ShapeFactory.getCircle(getRandomColor());
         circle.setX(getRandomX());
         circle.setY(getRandomY());
         circle.setRadius(100);
         circle.draw();
      }
   }
   private static String getRandomColor() {
      return colors[(int)(Math.random()*colors.length)];
   }
   private static int getRandomX() {
      return (int)(Math.random()*100 );
   }
   private static int getRandomY() {
      return (int)(Math.random()*100);
   }
}

ステップ5

出力を確認します。

Creating circle of color : Black
Circle: Draw() [Color : Black, x : 36, y :71, radius :100
Creating circle of color : Green
Circle: Draw() [Color : Green, x : 27, y :27, radius :100
Creating circle of color : White
Circle: Draw() [Color : White, x : 64, y :10, radius :100
Creating circle of color : Red
Circle: Draw() [Color : Red, x : 15, y :44, radius :100
Circle: Draw() [Color : Green, x : 19, y :10, radius :100
Circle: Draw() [Color : Green, x : 94, y :32, radius :100
Circle: Draw() [Color : White, x : 69, y :98, radius :100
Creating circle of color : Blue
Circle: Draw() [Color : Blue, x : 13, y :4, radius :100
Circle: Draw() [Color : Green, x : 21, y :21, radius :100
Circle: Draw() [Color : Blue, x : 55, y :86, radius :100
Circle: Draw() [Color : White, x : 90, y :70, radius :100
Circle: Draw() [Color : Green, x : 78, y :3, radius :100
Circle: Draw() [Color : Green, x : 64, y :89, radius :100
Circle: Draw() [Color : Blue, x : 3, y :91, radius :100
Circle: Draw() [Color : Blue, x : 62, y :82, radius :100
Circle: Draw() [Color : Green, x : 97, y :61, radius :100
Circle: Draw() [Color : Green, x : 86, y :12, radius :100
Circle: Draw() [Color : Green, x : 38, y :93, radius :100
Circle: Draw() [Color : Red, x : 76, y :82, radius :100
Circle: Draw() [Color : Blue, x : 95, y :82, radius :100

プロキシパターンでは、クラスは別のクラスの機能を表します。このタイプのデザインパターンは、構造パターンに分類されます。

プロキシパターンでは、元のオブジェクトを持つオブジェクトを作成して、その機能を外界にインターフェースします。

実装

私たちは、作成しようとしているイメージ・インタフェースと実装する具象クラスイメージ・インターフェイスを。ProxyImageは、RealImageオブジェクトの読み込みのメモリフットプリントを削減するためのプロキシクラスです。

ProxyPatternDemo、デモクラスはProxyImageを使用して、必要に応じてロードおよび表示するImageオブジェクトを取得します。

ステップ1

インターフェイスを作成します。

Image.java

public interface Image {
   void display();
}

ステップ2

同じインターフェースを実装する具象クラスを作成します。

RealImage.java

public class RealImage implements Image {

   private String fileName;

   public RealImage(String fileName){
      this.fileName = fileName;
      loadFromDisk(fileName);
   }

   @Override
   public void display() {
      System.out.println("Displaying " + fileName);
   }

   private void loadFromDisk(String fileName){
      System.out.println("Loading " + fileName);
   }
}

ProxyImage.java

public class ProxyImage implements Image{

   private RealImage realImage;
   private String fileName;

   public ProxyImage(String fileName){
      this.fileName = fileName;
   }

   @Override
   public void display() {
      if(realImage == null){
         realImage = new RealImage(fileName);
      }
      realImage.display();
   }
}

ステップ3

使用ProxyImageを対象取得するRealImageの必要な時にクラスを。

ProxyPatternDemo.java

public class ProxyPatternDemo {
	
   public static void main(String[] args) {
      Image image = new ProxyImage("test_10mb.jpg");

      //image will be loaded from disk
      image.display(); 
      System.out.println("");
      //image will not be loaded from disk
      image.display(); 	
   }
}

ステップ4

出力を確認します。

Loading test_10mb.jpg
Displaying test_10mb.jpg

Displaying test_10mb.jpg

名前が示すように、Chain of Responsibilityパターンは、リクエストのレシーバーオブジェクトのチェーンを作成します。このパターンは、リクエストのタイプに基づいて、リクエストの送信者と受信者を分離します。このパターンは行動パターンに分類されます。

このパターンでは、通常、各レシーバーには別のレシーバーへの参照が含まれています。1つのオブジェクトが要求を処理できない場合、次の受信者に同じものを渡します。

実装

ロギングのレベルを持つ抽象クラスAbstractLoggerを作成しました。次に、AbstractLoggerを拡張する3種類のロガーを作成しました。各ロガーはメッセージのレベルをそのレベルまでチェックし、それに応じて印刷します。そうでない場合、メッセージを印刷して次のロガーに渡しません。

ステップ1

抽象ロガークラスを作成します。

AbstractLogger.java

public abstract class AbstractLogger {
   public static int INFO = 1;
   public static int DEBUG = 2;
   public static int ERROR = 3;

   protected int level;

   //next element in chain or responsibility
   protected AbstractLogger nextLogger;

   public void setNextLogger(AbstractLogger nextLogger){
      this.nextLogger = nextLogger;
   }

   public void logMessage(int level, String message){
      if(this.level <= level){
         write(message);
      }
      if(nextLogger !=null){
         nextLogger.logMessage(level, message);
      }
   }

   abstract protected void write(String message);
	
}

ステップ2

ロガーを拡張する具象クラスを作成します。

ConsoleLogger.java

public class ConsoleLogger extends AbstractLogger {

   public ConsoleLogger(int level){
      this.level = level;
   }

   @Override
   protected void write(String message) {		
      System.out.println("Standard Console::Logger: " + message);
   }
}

ErrorLogger.java

public class ErrorLogger extends AbstractLogger {

   public ErrorLogger(int level){
      this.level = level;
   }

   @Override
   protected void write(String message) {		
      System.out.println("Error Console::Logger: " + message);
   }
}

FileLogger.java

public class FileLogger extends AbstractLogger {

   public FileLogger(int level){
      this.level = level;
   }

   @Override
   protected void write(String message) {		
      System.out.println("File::Logger: " + message);
   }
}

ステップ3

さまざまな種類のロガーを作成します。それらにエラーレベルを割り当て、各ロガーに次のロガーを設定します。各ロガーの次のロガーは、チェーンの一部を表します。

ChainPatternDemo.java

public class ChainPatternDemo {
	
   private static AbstractLogger getChainOfLoggers(){

      AbstractLogger errorLogger = new ErrorLogger(AbstractLogger.ERROR);
      AbstractLogger fileLogger = new FileLogger(AbstractLogger.DEBUG);
      AbstractLogger consoleLogger = new ConsoleLogger(AbstractLogger.INFO);

      errorLogger.setNextLogger(fileLogger);
      fileLogger.setNextLogger(consoleLogger);

      return errorLogger;	
   }

   public static void main(String[] args) {
      AbstractLogger loggerChain = getChainOfLoggers();

      loggerChain.logMessage(AbstractLogger.INFO, 
         "This is an information.");

      loggerChain.logMessage(AbstractLogger.DEBUG, 
         "This is an debug level information.");

      loggerChain.logMessage(AbstractLogger.ERROR, 
         "This is an error information.");
   }
}

ステップ4

出力を確認します。

Standard Console::Logger: This is an information.
File::Logger: This is an debug level information.
Standard Console::Logger: This is an debug level information.
Error Console::Logger: This is an error information.
File::Logger: This is an error information.
Standard Console::Logger: This is an error information.

コマンドパターンはデータ駆動型のデザインパターンであり、動作パターンのカテゴリに分類されます。リクエストはコマンドとしてオブジェクトの下にラップされ、呼び出し元オブジェクトに渡されます。呼び出し元オブジェクトは、このコマンドを処理できる適切なオブジェクトを探し、対応するオブジェクトにコマンドを渡すと、そのオブジェクトがコマンドを実行します。

実装

コマンドとして機能するインターフェイスOrderを作成しました。リクエストとして機能するStockクラスを作成しました。実際のコマンド処理を行うOrderインターフェイスを実装する具体的なコマンドクラスBuyStockSellStockがあります。呼び出し元オブジェクトとして機能するクラスブローカーが作成されます。それは注文を受けて注文することができます。

ブローカオブジェクトは、コマンドパターンを使用して、コマンドのタイプに基づいて、どのオブジェクトがどのコマンドを実行するかを識別します。CommandPatternDemo、デモクラスはBrokerクラスを使用してコマンドパターンを示します。

ステップ1

コマンドインターフェイスを作成します。

Order.java

public interface Order {
   void execute();
}

ステップ2

リクエストクラスを作成します。

Stock.java

public class Stock {
	
   private String name = "ABC";
   private int quantity = 10;

   public void buy(){
      System.out.println("Stock [ Name: "+name+", 
         Quantity: " + quantity +" ] bought");
   }
   public void sell(){
      System.out.println("Stock [ Name: "+name+", 
         Quantity: " + quantity +" ] sold");
   }
}

ステップ3

Orderインターフェースを実装する具象クラスを作成します。

BuyStock.java

public class BuyStock implements Order {
   private Stock abcStock;

   public BuyStock(Stock abcStock){
      this.abcStock = abcStock;
   }

   public void execute() {
      abcStock.buy();
   }
}

SellStock.java

public class SellStock implements Order {
   private Stock abcStock;

   public SellStock(Stock abcStock){
      this.abcStock = abcStock;
   }

   public void execute() {
      abcStock.sell();
   }
}

ステップ4

コマンド呼び出しクラスを作成します。

Broker.java

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

   public class Broker {
   private List<Order> orderList = new ArrayList<Order>(); 

   public void takeOrder(Order order){
      orderList.add(order);		
   }

   public void placeOrders(){
      for (Order order : orderList) {
         order.execute();
      }
      orderList.clear();
   }
}

ステップ5

Brokerクラスを使用して、コマンドを取得して実行します。

CommandPatternDemo.java

public class CommandPatternDemo {
   public static void main(String[] args) {
      Stock abcStock = new Stock();

      BuyStock buyStockOrder = new BuyStock(abcStock);
      SellStock sellStockOrder = new SellStock(abcStock);

      Broker broker = new Broker();
      broker.takeOrder(buyStockOrder);
      broker.takeOrder(sellStockOrder);

      broker.placeOrders();
   }
}

ステップ6

出力を確認します。

Stock [ Name: ABC, Quantity: 10 ] bought
Stock [ Name: ABC, Quantity: 10 ] sold

インタープリターパターンは、言語の文法または表現を評価する方法を提供します。このタイプのパターンは、行動パターンに分類されます。このパターンには、特定のコンテキストを解釈するように指示する式インターフェイスの実装が含まれます。このパターンは、SQL解析、シンボル処理エンジンなどで使用されます。

実装

インターフェースExpressionExpressionインターフェースを実装する具象クラスを作成します。問題のコンテキストのメインインタープリターとして機能するクラスTerminalExpressionが定義されています。他のクラスOrExpressionAndExpressionは、組み合わせ式を作成するために使用されます。

InterpreterPatternDemo、デモクラスはExpressionクラスを使用してルールを作成し、式の解析を示します。

ステップ1

式インターフェイスを作成します。

Expression.java

public interface Expression {
   public boolean interpret(String context);
}

ステップ2

上記のインターフェースを実装する具象クラスを作成します。

TerminalExpression.java

public class TerminalExpression implements Expression {
	
   private String data;

   public TerminalExpression(String data){
      this.data = data; 
   }

   @Override
   public boolean interpret(String context) {
      if(context.contains(data)){
         return true;
      }
      return false;
   }
}

OrExpression.java

public class OrExpression implements Expression {
	 
   private Expression expr1 = null;
   private Expression expr2 = null;

   public OrExpression(Expression expr1, Expression expr2) { 
      this.expr1 = expr1;
      this.expr2 = expr2;
   }

   @Override
   public boolean interpret(String context) {		
      return expr1.interpret(context) || expr2.interpret(context);
   }
}

AndExpression.java

public class AndExpression implements Expression {
	 
   private Expression expr1 = null;
   private Expression expr2 = null;

   public AndExpression(Expression expr1, Expression expr2) { 
      this.expr1 = expr1;
      this.expr2 = expr2;
   }

   @Override
   public boolean interpret(String context) {		
      return expr1.interpret(context) && expr2.interpret(context);
   }
}

ステップ3

InterpreterPatternDemoは、Expressionクラスを使用してルールを作成し、それらを解析します。

InterpreterPatternDemo.java

public class InterpreterPatternDemo {

   //Rule: Robert and John are male
   public static Expression getMaleExpression(){
      Expression robert = new TerminalExpression("Robert");
      Expression john = new TerminalExpression("John");
      return new OrExpression(robert, john);		
   }

   //Rule: Julie is a married women
   public static Expression getMarriedWomanExpression(){
      Expression julie = new TerminalExpression("Julie");
      Expression married = new TerminalExpression("Married");
      return new AndExpression(julie, married);		
   }

   public static void main(String[] args) {
      Expression isMale = getMaleExpression();
      Expression isMarriedWoman = getMarriedWomanExpression();

      System.out.println("John is male? " + isMale.interpret("John"));
      System.out.println("Julie is a married women? " 
      + isMarriedWoman.interpret("Married Julie"));
   }
}

ステップ4

出力を確認します。

John is male? true
Julie is a married women? true

Iteratorパターンは、Javaおよび.Netプログラミング環境で非常に一般的に使用されるデザインパターンです。このパターンは、基礎となる表現を知る必要なしに、コレクションオブジェクトの要素に順番にアクセスする方法を取得するために使用されます。

イテレーターパターンは、行動パターンのカテゴリに分類されます。

実装

ナビゲーションメソッドをナレーションするIteratorインターフェイスと、Iteratorを再実行するContainerインターフェイスを作成します。Containerインターフェースを実装する具象クラスは、Iteratorインターフェースを実装して使用する責任があります。

IteratorPatternDemo、デモクラスはNamesRepositoryを使用します。これは、NamesRepositoryにコレクションとして格納されているNamesを出力するための具体的なクラス実装です

ステップ1

インターフェイスを作成します。

Iterator.java

public interface Iterator {
   public boolean hasNext();
   public Object next();
}

Container.java

public interface Container {
   public Iterator getIterator();
}

ステップ2

Containerインターフェースを実装する具象クラスを作成します。このクラスには、Iteratorインターフェースを実装する内部クラスNameIteratorがあります。

NameRepository.java

public class NameRepository implements Container {
   public String names[] = {"Robert" , "John" ,"Julie" , "Lora"};

   @Override
   public Iterator getIterator() {
      return new NameIterator();
   }

   private class NameIterator implements Iterator {

      int index;

      @Override
      public boolean hasNext() {
         if(index < names.length){
            return true;
         }
         return false;
      }

      @Override
      public Object next() {
         if(this.hasNext()){
            return names[index++];
         }
         return null;
      }		
   }
}

ステップ3

NameRepositoryを使用して、イテレーターを取得し、名前を出力します。

IteratorPatternDemo.java

public class IteratorPatternDemo {
	
   public static void main(String[] args) {
      NameRepository namesRepository = new NameRepository();

      for(Iterator iter = namesRepository.getIterator(); iter.hasNext();){
         String name = (String)iter.next();
         System.out.println("Name : " + name);
      } 	
   }
}

ステップ4

出力を確認します。

Name : Robert
Name : John
Name : Julie
Name : Lora

メディエーターパターンは、複数のオブジェクトまたはクラス間の通信の複雑さを軽減するために使用されます。このパターンは、通常、異なるクラス間のすべての通信を処理し、疎結合によってコードの保守性を容易にするメディエータークラスを提供します。メディエーターパターンは、行動パターンのカテゴリーに分類されます。

実装

複数のユーザーがチャットルームにメッセージを送信できるチャットルームの例でメディエーターパターンを示しています。すべてのユーザーにメッセージを表示するのはチャットルームの責任です。ChatRoomUserの2つのクラスを作成しました。ユーザーオブジェクトは、ChatRoomメソッドを使用してメッセージを共有します。

MediatorPatternDemo、デモクラスはUserオブジェクトを使用して、それらの間の通信を表示します。

ステップ1

メディエータークラスを作成します。

ChatRoom.java

import java.util.Date;

public class ChatRoom {
   public static void showMessage(User user, String message){
      System.out.println(new Date().toString()
         + " [" + user.getName() +"] : " + message);
   }
}

ステップ2

ユーザークラスを作成する

User.java

public class User {
   private String name;

   public String getName() {
      return name;
   }

   public void setName(String name) {
      this.name = name;
   }

   public User(String name){
      this.name  = name;
   }

   public void sendMessage(String message){
      ChatRoom.showMessage(this,message);
   }
}

ステップ3

Userオブジェクトを使用して、それらの間の通信を表示します。

MediatorPatternDemo.java

public class MediatorPatternDemo {
   public static void main(String[] args) {
      User robert = new User("Robert");
      User john = new User("John");

      robert.sendMessage("Hi! John!");
      john.sendMessage("Hello! Robert!");
   }
}

ステップ4

出力を確認します。

Thu Jan 31 16:05:46 IST 2013 [Robert] : Hi! John!
Thu Jan 31 16:05:46 IST 2013 [John] : Hello! Robert!

Mementoパターンは、オブジェクトの状態を以前の状態に復元する場所を減らすために使用されます。Mementoパターンは、行動パターンのカテゴリに分類されます。

実装

Mementoパターンは3つのアクタークラスを使用します。Mementoには、復元するオブジェクトの状態が含まれています。オリジネーターは、MementoオブジェクトとMementoからオブジェクトの状態を復元する責任があるCaretakerオブジェクトに状態を作成して保存します。MementoOriginatorCareTakerのクラスを作成しました。

MementoPatternDemo、デモクラスは、CareTakerオブジェクトとOriginatorオブジェクトを使用して、オブジェクトの状態の復元を示します。

ステップ1

Mementoクラスを作成します。

Memento.java

public class Memento {
   private String state;

   public Memento(String state){
      this.state = state;
   }

   public String getState(){
      return state;
   }	
}

ステップ2

オリジネータークラスを作成する

Originator.java

public class Originator {
   private String state;

   public void setState(String state){
      this.state = state;
   }

   public String getState(){
      return state;
   }

   public Memento saveStateToMemento(){
      return new Memento(state);
   }

   public void getStateFromMemento(Memento Memento){
      state = memento.getState();
   }
}

ステップ3

CareTakerクラスを作成する

CareTaker.java

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

public class CareTaker {
   private List<Memento> mementoList = new ArrayList<Memento>();

   public void add(Memento state){
      mementoList.add(state);
   }

   public Memento get(int index){
      return mementoList.get(index);
   }
}

ステップ4

CareTakerオブジェクトとOriginatorオブジェクトを使用します。

MementoPatternDemo.java

public class MementoPatternDemo {
   public static void main(String[] args) {
      Originator originator = new Originator();
      CareTaker careTaker = new CareTaker();
      originator.setState("State #1");
      originator.setState("State #2");
      careTaker.add(originator.saveStateToMemento());
      originator.setState("State #3");
      careTaker.add(originator.saveStateToMemento());
      originator.setState("State #4");

      System.out.println("Current State: " + originator.getState());		
      originator.getStateFromMemento(careTaker.get(0));
      System.out.println("First saved State: " + originator.getState());
      originator.getStateFromMemento(careTaker.get(1));
      System.out.println("Second saved State: " + originator.getState());
   }
}

ステップ5

出力を確認します。

Current State: State #4
First saved State: State #2
Second saved State: State #3

オブザーバーパターンは、1つのオブジェクトが変更された場合など、オブジェクト間に1対多の関係がある場合に使用され、その依存オブジェクトは自動的に通知されます。オブザーバーパターンは、行動パターンのカテゴリに分類されます。

実装

オブザーバーパターンは3つのアクタークラスを使用します。サブジェクト、オブザーバー、クライアント。サブジェクト、オブザーバーをクライアントオブジェクトにアタッチおよびデアタッチするメソッドを持つオブジェクト。私たちは、クラスの作成した件名オブザーバー抽象クラスと抽象クラス拡張した具体クラスオブザーバーを

ObserverPatternDemo、デモクラスは、Subjectクラスオブジェクトと具象クラスオブジェクトを使用して、動作中のオブザーバーパターンを示します。

ステップ1

サブジェクトクラスを作成します。

Subject.java

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

public class Subject {
	
   private List<Observer> observers 
      = new ArrayList<Observer>();
   private int state;

   public int getState() {
      return state;
   }

   public void setState(int state) {
      this.state = state;
      notifyAllObservers();
   }

   public void attach(Observer observer){
      observers.add(observer);		
   }

   public void notifyAllObservers(){
      for (Observer observer : observers) {
         observer.update();
      }
   } 	
}

ステップ2

オブザーバークラスを作成します。

Observer.java

public abstract class Observer {
   protected Subject subject;
   public abstract void update();
}

ステップ3

具体的なオブザーバークラスを作成する

BinaryObserver.java

public class BinaryObserver extends Observer{

   public BinaryObserver(Subject subject){
      this.subject = subject;
      this.subject.attach(this);
   }

   @Override
   public void update() {
      System.out.println( "Binary String: " 
      + Integer.toBinaryString( subject.getState() ) ); 
   }
}

OctalObserver.java

public class OctalObserver extends Observer{

   public OctalObserver(Subject subject){
      this.subject = subject;
      this.subject.attach(this);
   }

   @Override
   public void update() {
     System.out.println( "Octal String: " 
     + Integer.toOctalString( subject.getState() ) ); 
   }
}

HexaObserver.java

public class HexaObserver extends Observer{

   public HexaObserver(Subject subject){
      this.subject = subject;
      this.subject.attach(this);
   }

   @Override
   public void update() {
      System.out.println( "Hex String: " 
      + Integer.toHexString( subject.getState() ).toUpperCase() ); 
   }
}

ステップ4

サブジェクトオブジェクトと具象オブザーバーオブジェクトを使用します。

ObserverPatternDemo.java

public class ObserverPatternDemo {
   public static void main(String[] args) {
      Subject subject = new Subject();

      new HexaObserver(subject);
      new OctalObserver(subject);
      new BinaryObserver(subject);

      System.out.println("First state change: 15");	
      subject.setState(15);
      System.out.println("Second state change: 10");	
      subject.setState(10);
   }
}

ステップ5

出力を確認します。

First state change: 15
Hex String: F
Octal String: 17
Binary String: 1111
Second state change: 10
Hex String: A
Octal String: 12
Binary String: 1010

状態パターンでは、クラスの動作はその状態に基づいて変化します。このタイプのデザインパターンは、動作パターンに分類されます。

状態パターンでは、さまざまな状態を表すオブジェクトと、状態オブジェクトの変化に応じて動作が変化するコンテキストオブジェクトを作成します。

実装

アクションを定義するStateインターフェースと、Stateインターフェースを実装する具象状態クラスを作成します。コンテキストは、状態を運ぶクラスです。

StaePatternDemo、デモクラスは、Contextオブジェクトとstateオブジェクトを使用して、状態のタイプに基づいたContextの動作の変化を示します。

ステップ1

インターフェイスを作成します。

Image.java

public interface State {
   public void doAction(Context context);
}

ステップ2

同じインターフェースを実装する具象クラスを作成します。

StartState.java

public class StartState implements State {

   public void doAction(Context context) {
      System.out.println("Player is in start state");
      context.setState(this);	
   }

   public String toString(){
      return "Start State";
   }
}

StopState.java

public class StopState implements State {

   public void doAction(Context context) {
      System.out.println("Player is in stop state");
      context.setState(this);	
   }

   public String toString(){
      return "Stop State";
   }
}

ステップ3

コンテキストクラスを作成します。

Context.java

public class Context {
   private State state;

   public Context(){
      state = null;
   }

   public void setState(State state){
      this.state = state;		
   }

   public State getState(){
      return state;
   }
}

ステップ4

コンテキストを使用して、状態が変化したときの動作の変化を確認します。

StatePatternDemo.java

public class StatePatternDemo {
   public static void main(String[] args) {
      Context context = new Context();

      StartState startState = new StartState();
      startState.doAction(context);

      System.out.println(context.getState().toString());

      StopState stopState = new StopState();
      stopState.doAction(context);

      System.out.println(context.getState().toString());
   }
}

ステップ5

出力を確認します。

Player is in start state
Start State
Player is in stop state
Stop State

ヌルオブジェクトパターンでは、ヌルオブジェクトがNULLオブジェクトインスタンスのチェックに置き換わります。null値をチェックするかどうかを指定する代わりに、Nullオブジェクトは何もしない関係を反映します。このようなNullオブジェクトを使用して、データが利用できない場合のデフォルトの動作を提供することもできます。

ヌルオブジェクトパターンでは、実行するさまざまな操作を指定する抽象クラスを作成し、このクラスを拡張するクラスと、このクラスの実装を何も提供しないヌルオブジェクトクラスを作成し、ヌル値をチェックする必要がある場合に使用されます。

実装

操作を定義するAbstractCustomer抽象クラスを作成します。ここでは、顧客の名前と、AbstractCustomerクラスを拡張する具象クラスを作成します。ファクトリクラスCustomerFactoryはどちらかを返すために作成されRealCustomerまたはNullCustomerがそれに渡された顧客の名前に基づいてオブジェクト。

NullPatternDemo、デモクラスはCustomerFactoryを使用して、NullObjectパターンの使用法を示します。

ステップ1

抽象クラスを作成します。

AbstractCustomer.java

public abstract class AbstractCustomer {
   protected String name;
   public abstract boolean isNil();
   public abstract String getName();
}

ステップ2

上記のクラスを拡張する具象クラスを作成します。

RealCustomer.java

public class RealCustomer extends AbstractCustomer {

   public RealCustomer(String name) {
      this.name = name;		
   }
   
   @Override
   public String getName() {
      return name;
   }
   
   @Override
   public boolean isNil() {
      return false;
   }
}

NullCustomer.java

public class NullCustomer extends AbstractCustomer {

   @Override
   public String getName() {
      return "Not Available in Customer Database";
   }

   @Override
   public boolean isNil() {
      return true;
   }
}

ステップ3

CustomerFactoryクラスを作成します。

CustomerFactory.java

public class CustomerFactory {
	
   public static final String[] names = {"Rob", "Joe", "Julie"};

   public static AbstractCustomer getCustomer(String name){
      for (int i = 0; i < names.length; i++) {
         if (names[i].equalsIgnoreCase(name)){
            return new RealCustomer(name);
         }
      }
      return new NullCustomer();
   }
}

ステップ4

使用CustomerFactory GETのいずれかRealCustomerまたはNullCustomerがそれに渡された顧客の名前に基づいてオブジェクト。

NullPatternDemo.java

public class NullPatternDemo {
   public static void main(String[] args) {

      AbstractCustomer customer1 = CustomerFactory.getCustomer("Rob");
      AbstractCustomer customer2 = CustomerFactory.getCustomer("Bob");
      AbstractCustomer customer3 = CustomerFactory.getCustomer("Julie");
      AbstractCustomer customer4 = CustomerFactory.getCustomer("Laura");

      System.out.println("Customers");
      System.out.println(customer1.getName());
      System.out.println(customer2.getName());
      System.out.println(customer3.getName());
      System.out.println(customer4.getName());
   }
}

ステップ5

出力を確認します。

Customers
Rob
Not Available in Customer Database
Julie
Not Available in Customer Database

ストラテジーパターンでは、クラスの動作またはそのアルゴリズムを実行時に変更できます。このタイプのデザインパターンは、動作パターンに分類されます。

ストラテジーパターンでは、さまざまなストラテジーを表すオブジェクトと、ストラテジーオブジェクトごとに動作が異なるコンテキストオブジェクトを作成します。ストラテジーオブジェクトは、コンテキストオブジェクトの実行アルゴリズムを変更します。

実装

アクションを定義するStrategyインターフェースと、Strategyインターフェースを実装する具体的な戦略クラスを作成します。コンテキストは、ストラテジーを使用するクラスです。

StrategyPatternDemo、デモクラスは、Contextオブジェクトとstrategyオブジェクトを使用して、デプロイまたは使用する戦略に基づいたContextの動作の変化を示します。

ステップ1

インターフェイスを作成します。

Strategy.java

public interface Strategy {
   public int doOperation(int num1, int num2);
}

ステップ2

同じインターフェースを実装する具象クラスを作成します。

OperationAdd.java

public class OperationAdd implements Strategy{
   @Override
   public int doOperation(int num1, int num2) {
      return num1 + num2;
   }
}

OperationSubstract.java

public class OperationSubstract implements Strategy{
   @Override
   public int doOperation(int num1, int num2) {
      return num1 - num2;
   }
}

OperationMultiply.java

public class OperationMultiply implements Strategy{
   @Override
   public int doOperation(int num1, int num2) {
      return num1 * num2;
   }
}

ステップ3

コンテキストクラスを作成します。

Context.java

public class Context {
   private Strategy strategy;

   public Context(Strategy strategy){
      this.strategy = strategy;
   }

   public int executeStrategy(int num1, int num2){
      return strategy.doOperation(num1, num2);
   }
}

ステップ4

コンテキストを使用して、戦略を変更したときの動作の変化を確認します。

StatePatternDemo.java

public class StrategyPatternDemo {
   public static void main(String[] args) {
      Context context = new Context(new OperationAdd());		
      System.out.println("10 + 5 = " + context.executeStrategy(10, 5));

      context = new Context(new OperationSubstract());		
      System.out.println("10 - 5 = " + context.executeStrategy(10, 5));

      context = new Context(new OperationMultiply());		
      System.out.println("10 * 5 = " + context.executeStrategy(10, 5));
   }
}

ステップ5

出力を確認します。

10 + 5 = 15
10 - 5 = 5
10 * 5 = 50

テンプレートパターンでは、抽象クラスはそのメソッドを実行するために定義された方法/テンプレートを公開します。そのサブクラスは、必要に応じてメソッドの実装をオーバーライドできますが、呼び出しは抽象クラスで定義されているのと同じ方法で行う必要があります。このパターンは、行動パターンのカテゴリに分類されます。

実装

テンプレートメソッドをfinalに設定して操作を定義するGame抽象クラスを作成し、オーバーライドできないようにします。クリケットサッカーは具体的なクラスであり、ゲームを拡張し、そのメソッドをオーバーライドします。

TemplatePatternDemo、私たちのデモクラスはGameを使用してテンプレートパターンの使用法を示します。

ステップ1

最終的なテンプレートメソッドを使用して抽象クラスを作成します。

Game.java

public abstract class Game {
   abstract void initialize();
   abstract void startPlay();
   abstract void endPlay();

   //template method
   public final void play(){

      //initialize the game
      initialize();

      //start game
      startPlay();

      //end game
      endPlay();
   }
}

ステップ2

上記のクラスを拡張する具象クラスを作成します。

Cricket.java

public class Cricket extends Game {

   @Override
   void endPlay() {
      System.out.println("Cricket Game Finished!");
   }

   @Override
   void initialize() {
      System.out.println("Cricket Game Initialized! Start playing.");
   }

   @Override
   void startPlay() {
      System.out.println("Cricket Game Started. Enjoy the game!");
   }
}

Football.java

public class Football extends Game {
   @Override
   void endPlay() {
      System.out.println("Football Game Finished!");
   }

   @Override
   void initialize() {
      System.out.println("Football Game Initialized! Start playing.");
   }

   @Override
   void startPlay() {
      System.out.println("Football Game Started. Enjoy the game!");
   }
}

ステップ3

ゲームのテンプレートメソッドplay()を使用して、定義されたゲームのプレイ方法を示します。

TemplatePatternDemo.java

public class TemplatePatternDemo {
   public static void main(String[] args) {

      Game game = new Cricket();
      game.play();
      System.out.println();
      game = new Football();
      game.play();		
   }
}

ステップ4

出力を確認します。

Cricket Game Initialized! Start playing.
Cricket Game Started. Enjoy the game!
Cricket Game Finished!

Football Game Initialized! Start playing.
Football Game Started. Enjoy the game!
Football Game Finished!

ビジターパターンでは、エレメントクラスの実行アルゴリズムを変更するビジタークラスを使用します。このように、要素の実行アルゴリズムは、訪問者の変化に応じて変化する可能性があります。このパターンは、行動パターンのカテゴリに分類されます。パターンに従って、訪問者オブジェクトが要素オブジェクトに対する操作を処理できるように、要素オブジェクトは訪問者オブジェクトを受け入れる必要があります。

実装

受け入れ操作を定義するComputerPartインターフェイスを作成します。KeyboardMouseMonitorComputerは、ComputerPartインターフェイスを実装する具体的なクラスです。ビジタークラスの操作を定義する別のインターフェイスComputerPartVisitorを定義します。コンピューターは具体的な訪問者を使用して対応するアクションを実行します。

VisitorPatternDemo、私たちのデモクラスは、ComputerComputerPartVisitorクラスを使用して、ビジターパターンの使用法を示します。

ステップ1

要素を表すインターフェイスを定義します。

ComputerPart.java

public interface class ComputerPart {
   public void accept(ComputerPartVisitor computerPartVisitor);
}

ステップ2

上記のクラスを拡張する具象クラスを作成します。

Keyboard.java

public class Keyboard  implements ComputerPart {

   @Override
   public void accept(ComputerPartVisitor computerPartVisitor) {
      computerPartVisitor.visit(this);
   }
}

Monitor.java

public class Monitor  implements ComputerPart {

   @Override
   public void accept(ComputerPartVisitor computerPartVisitor) {
      computerPartVisitor.visit(this);
   }
}

Mouse.java

public class Mouse  implements ComputerPart {

   @Override
   public void accept(ComputerPartVisitor computerPartVisitor) {
      computerPartVisitor.visit(this);
   }
}

Computer.java

public class Computer implements ComputerPart {
	
   ComputerPart[] parts;

   public Computer(){
      parts = new ComputerPart[] {new Mouse(), new Keyboard(), new Monitor()};		
   } 


   @Override
   public void accept(ComputerPartVisitor computerPartVisitor) {
      for (int i = 0; i < parts.length; i++) {
         parts[i].accept(computerPartVisitor);
      }
      computerPartVisitor.visit(this);
   }
}

ステップ3

訪問者を表すインターフェイスを定義します。

ComputerPartVisitor.java

public interface ComputerPartVisitor {
	public void visit(Computer computer);
	public void visit(Mouse mouse);
	public void visit(Keyboard keyboard);
	public void visit(Monitor monitor);
}

ステップ4

上記のクラスを実装する具体的な訪問者を作成します。

ComputerPartDisplayVisitor.java

public class ComputerPartDisplayVisitor implements ComputerPartVisitor {

   @Override
   public void visit(Computer computer) {
      System.out.println("Displaying Computer.");
   }

   @Override
   public void visit(Mouse mouse) {
      System.out.println("Displaying Mouse.");
   }

   @Override
   public void visit(Keyboard keyboard) {
      System.out.println("Displaying Keyboard.");
   }

   @Override
   public void visit(Monitor monitor) {
      System.out.println("Displaying Monitor.");
   }
}

ステップ5

使用ComputerPartDisplayVisitorをする部品表示するコンピュータを

VisitorPatternDemo.java

public class VisitorPatternDemo {
   public static void main(String[] args) {

      ComputerPart computer = new Computer();
      computer.accept(new ComputerPartDisplayVisitor());
   }
}

ステップ6

出力を確認します。

Displaying Mouse.
Displaying Keyboard.
Displaying Monitor.
Displaying Computer.

MVCパターンはModel-View-ControllerPatternの略です。このパターンは、アプリケーションの懸念を分離するために使用されます。

  • Model-モデルは、データを運ぶオブジェクトまたはJAVAPOJOを表します。また、データが変更された場合にコントローラーを更新するロジックを持つこともできます。

  • View -ビューは、モデルに含まれるデータの視覚化を表します。

  • Controller-コントローラーはモデルとビューの両方に作用します。モデルオブジェクトへのデータフローを制御し、データが変更されるたびにビューを更新します。ビューとモデルを分離します。

実装

モデルとして機能するStudentオブジェクトを作成します。StudentViewは、コンソールに学生の詳細を出力できるビュークラスであり、StudentControllerは、Studentオブジェクトにデータを格納し、それに応じてビューStudentViewを更新するコントローラークラスです。

MVCPatternDemo、デモクラスはStudentControllerを使用してMVCパターンの使用を示します。

ステップ1

モデルを作成します。

Student.java

public class Student {
   private String rollNo;
   private String name;
   public String getRollNo() {
      return rollNo;
   }
   public void setRollNo(String rollNo) {
      this.rollNo = rollNo;
   }
   public String getName() {
      return name;
   }
   public void setName(String name) {
      this.name = name;
   }
}

ステップ2

ビューを作成します。

StudentView.java

public class StudentView {
   public void printStudentDetails(String studentName, String studentRollNo){
      System.out.println("Student: ");
      System.out.println("Name: " + studentName);
      System.out.println("Roll No: " + studentRollNo);
   }
}

ステップ3

コントローラを作成します。

StudentController.java

public class StudentController {
   private Student model;
   private StudentView view;

   public StudentController(Student model, StudentView view){
      this.model = model;
      this.view = view;
   }

   public void setStudentName(String name){
      model.setName(name);		
   }

   public String getStudentName(){
      return model.getName();		
   }

   public void setStudentRollNo(String rollNo){
      model.setRollNo(rollNo);		
   }

   public String getStudentRollNo(){
      return model.getRollNo();		
   }

   public void updateView(){				
      view.printStudentDetails(model.getName(), model.getRollNo());
   }	
}

ステップ4

StudentControllerメソッドを使用して、MVCデザインパターンの使用法を示します。

MVCPatternDemo.java

public class MVCPatternDemo {
   public static void main(String[] args) {

      //fetch student record based on his roll no from the database
      Student model  = retriveStudentFromDatabase();

      //Create a view : to write student details on console
      StudentView view = new StudentView();

      StudentController controller = new StudentController(model, view);

      controller.updateView();

      //update model data
      controller.setStudentName("John");

      controller.updateView();
   }

   private static Student retriveStudentFromDatabase(){
      Student student = new Student();
      student.setName("Robert");
      student.setRollNo("10");
      return student;
   }
}

ステップ5

出力を確認します。

Student: 
Name: Robert
Roll No: 10
Student: 
Name: Julie
Roll No: 10

Business Delegate Patternは、プレゼンテーション層とビジネス層を分離するために使用されます。これは基本的に、通信またはリモートルックアップ機能をプレゼンテーション層コードのビジネス層コードに減らすために使用されます。ビジネス層では、次のエンティティがあります。

  • Client -プレゼンテーション層コードは、JSP、サーブレット、またはUIJavaコードの場合があります。

  • Business Delegate -クライアントエンティティがビジネスサービスメソッドへのアクセスを提供するための単一のエントリポイントクラス。

  • LookUp Service -ルックアップサービスオブジェクトは、相対的なビジネス実装を取得し、ビジネスデリゲートオブジェクトへのビジネスオブジェクトアクセスを提供する責任があります。

  • Business Service-ビジネスサービスインターフェイス。具象クラスは、このビジネスサービスを実装して、実際のビジネス実装ロジックを提供します。

実装

Business Delegateパターンのさまざまなエンティティを表すClientBusinessDelegateBusinessServiceLookUpServiceJMSService、およびEJBServiceを作成します。

BusinessDelegatePatternDemo、デモクラスでは、BusinessDelegateClientを使用して、BusinessDelegateパターンの使用方法を示します。

ステップ1

BusinessServiceインターフェイスを作成します。

BusinessService.java

public interface BusinessService {
   public void doProcessing();
}

ステップ2

Concreateサービスクラスを作成します。

EJBService.java

public class EJBService implements BusinessService {

   @Override
   public void doProcessing() {
      System.out.println("Processing task by invoking EJB Service");
   }
}

JMSService.java

public class JMSService implements BusinessService {

   @Override
   public void doProcessing() {
      System.out.println("Processing task by invoking JMS Service");
   }
}

ステップ3

ビジネスルックアップサービスを作成します。

BusinessLookUp.java

public class BusinessLookUp {
   public BusinessService getBusinessService(String serviceType){
      if(serviceType.equalsIgnoreCase("EJB")){
         return new EJBService();
      }else {
         return new JMSService();
      }
   }
}

ステップ4

ビジネスデリゲートを作成します。

BusinessLookUp.java

public class BusinessDelegate {
   private BusinessLookUp lookupService = new BusinessLookUp();
   private BusinessService businessService;
   private String serviceType;

   public void setServiceType(String serviceType){
      this.serviceType = serviceType;
   }

   public void doTask(){
      businessService = lookupService.getBusinessService(serviceType);
      businessService.doProcessing();		
   }
}

ステップ5

クライアントを作成します。

Student.java

public class Client {
	
   BusinessDelegate businessService;

   public Client(BusinessDelegate businessService){
      this.businessService  = businessService;
   }

   public void doTask(){		
      businessService.doTask();
   }
}

ステップ6

BusinessDelegateクラスとClientクラスを使用して、BusinessDelegateパターンを示します。

BusinessDelegatePatternDemo.java

public class BusinessDelegatePatternDemo {
	
   public static void main(String[] args) {

      BusinessDelegate businessDelegate = new BusinessDelegate();
      businessDelegate.setServiceType("EJB");

      Client client = new Client(businessDelegate);
      client.doTask();

      businessDelegate.setServiceType("JMS");
      client.doTask();
   }
}

ステップ7

出力を確認します。

Processing task by invoking EJB Service
Processing task by invoking JMS Service

複合エンティティパターンは、EJB永続性メカニズムで使用されます。複合エンティティは、オブジェクトのグラフを表すEJBエンティティBeanです。複合エンティティが更新されると、内部依存オブジェクトBeanは、EJBエンティティBeanによって管理されるように自動的に更新されます。以下は、Composite EntityBeanの参加者です。

  • Composite Entity -プライマリエンティティBeanです。粗粒度にすることも、永続化の目的で使用する粗粒度オブジェクトを含めることもできます。

  • Coarse-Grained Object-このオブジェクトには、依存オブジェクトが含まれています。独自のライフサイクルがあり、依存オブジェクトのライフサイクルも管理します。

  • Dependent Object -依存オブジェクトは、永続性のライフサイクルをCoarse-Grainedオブジェクトに依存するオブジェクトです。

  • Strategies -戦略は、複合エンティティを実装する方法を表します。

実装

CompositeEntityとして機能するCompositeEntityオブジェクトを作成します。CoarseGrainedObjectは、依存オブジェクトを含むクラスになります。CompositeEntityPatternDemo、デモクラスはClientクラスを使用して、CompositeEntityパターンの使用法を示します。

ステップ1

依存オブジェクトを作成します。

DependentObject1.java

public class DependentObject1 {
	
   private String data;

   public void setData(String data){
      this.data = data; 
   } 

   public String getData(){
      return data;
   }
}

DependentObject2.java

public class DependentObject2 {
	
   private String data;

   public void setData(String data){
      this.data = data; 
   } 

   public String getData(){
      return data;
   }
}

ステップ2

粗視化オブジェクトを作成します。

CoarseGrainedObject.java

public class CoarseGrainedObject {
   DependentObject1 do1 = new DependentObject1();
   DependentObject2 do2 = new DependentObject2();

   public void setData(String data1, String data2){
      do1.setData(data1);
      do2.setData(data2);
   }

   public String[] getData(){
      return new String[] {do1.getData(),do2.getData()};
   }
}

ステップ3

複合エンティティを作成します。

CompositeEntity.java

public class CompositeEntity {
   private CoarseGrainedObject cgo = new CoarseGrainedObject();

   public void setData(String data1, String data2){
      cgo.setData(data1, data2);
   }

   public String[] getData(){
      return cgo.getData();
   }
}

ステップ4

複合エンティティを使用するクライアントクラスを作成します。

Client.java

public class Client {
   private CompositeEntity compositeEntity = new CompositeEntity();

   public void printData(){
      for (int i = 0; i < compositeEntity.getData().length; i++) {
         System.out.println("Data: " + compositeEntity.getData()[i]);
      }
   }

   public void setData(String data1, String data2){
      compositeEntity.setData(data1, data2);
   }
}

ステップ5

クライアントを使用して、複合エンティティのデザインパターンの使用法を示します。

CompositeEntityPatternDemo.java

public class CompositeEntityPatternDemo {
   public static void main(String[] args) {
       Client client = new Client();
       client.setData("Test", "Data");
       client.printData();
       client.setData("Second Test", "Data1");
       client.printData();
   }
}

ステップ6

出力を確認します。

Data: Test
Data: Data
Data: Second Test
Data: Data1

データアクセスオブジェクトパターンまたはDAOパターンは、APIまたは操作にアクセスする低レベルのデータを高レベルのビジネスサービスから分離するために使用されます。以下は、データアクセスオブジェクトパターンの参加者です。

  • Data Access Object Interface -このインターフェースは、モデルオブジェクトに対して実行される標準操作を定義します。

  • Data Access Object concrete class-このクラスは上記のインターフェースを実装します。このクラスは、データベース/ xmlまたはその他のストレージメカニズムであるデータソースからデータを取得する役割を果たします。

  • Model Object or Value Object -このオブジェクトは、DAOクラスを使用して取得したデータを格納するためのget / setメソッドを含む単純なPOJOです。

実装

モデルまたは値オブジェクトとして機能するStudentオブジェクトを作成します。StudentDaoはデータアクセスオブジェクトインターフェイスです。StudentDaoImplは、データアクセスオブジェクトインターフェイスを実装する具象クラスです。DaoPatternDemo、私たちのデモクラスはStudentDaoを使用してデータアクセスオブジェクトパターンの使用を示します。

ステップ1

値オブジェクトを作成します。

Student.java

public class Student {
   private String name;
   private int rollNo;

   Student(String name, int rollNo){
      this.name = name;
      this.rollNo = rollNo;
   }

   public String getName() {
      return name;
   }

   public void setName(String name) {
      this.name = name;
   }

   public int getRollNo() {
      return rollNo;
   }

   public void setRollNo(int rollNo) {
      this.rollNo = rollNo;
   }
}

ステップ2

データアクセスオブジェクトインターフェイスを作成します。

StudentDao.java

import java.util.List;

public interface StudentDao {
   public List<Student> getAllStudents();
   public Student getStudent(int rollNo);
   public void updateStudent(Student student);
   public void deleteStudent(Student student);
}

ステップ3

上記のインターフェースを実装するconcreateクラスを作成します。

StudentDaoImpl.java

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

public class StudentDaoImpl implements StudentDao {
	
   //list is working as a database
   List<Student> students;

   public StudentDaoImpl(){
      students = new ArrayList<Student>();
      Student student1 = new Student("Robert",0);
      Student student2 = new Student("John",1);
      students.add(student1);
      students.add(student2);		
   }
   @Override
   public void deleteStudent(Student student) {
      students.remove(student.getRollNo());
      System.out.println("Student: Roll No " + student.getRollNo() 
         +", deleted from database");
   }

   //retrive list of students from the database
   @Override
   public List<Student> getAllStudents() {
      return students;
   }

   @Override
   public Student getStudent(int rollNo) {
      return students.get(rollNo);
   }

   @Override
   public void updateStudent(Student student) {
      students.get(student.getRollNo()).setName(student.getName());
      System.out.println("Student: Roll No " + student.getRollNo() 
         +", updated in the database");
   }
}

ステップ4

StudentDaoを使用して、データアクセスオブジェクトパターンの使用法を示します。

CompositeEntityPatternDemo.java

public class DaoPatternDemo {
   public static void main(String[] args) {
      StudentDao studentDao = new StudentDaoImpl();

      //print all students
      for (Student student : studentDao.getAllStudents()) {
         System.out.println("Student: [RollNo : "
            +student.getRollNo()+", Name : "+student.getName()+" ]");
      }


      //update student
      Student student =studentDao.getAllStudents().get(0);
      student.setName("Michael");
      studentDao.updateStudent(student);

      //get the student
      studentDao.getStudent(0);
      System.out.println("Student: [RollNo : "
         +student.getRollNo()+", Name : "+student.getName()+" ]");		
   }
}

ステップ5

出力を確認します。

Student: [RollNo : 0, Name : Robert ]
Student: [RollNo : 1, Name : John ]
Student: Roll No 0, updated in the database
Student: [RollNo : 0, Name : Michael ]

フロントコントローラーのデザインパターンは、すべてのリクエストが単一のハンドラーによって処理されるように、一元化されたリクエスト処理メカニズムを提供するために使用されます。このハンドラーは、要求の認証/承認/ロギングまたは追跡を実行してから、対応するハンドラーに要求を渡すことができます。以下は、このタイプのデザインパターンのエンティティです。

  • Front Controller -アプリケーションに送信されるすべての種類の要求(Webベース/デスクトップベース)の単一ハンドラー。

  • Dispatcher -フロントコントローラーは、対応する特定のハンドラーにリクエストをディスパッチできるディスパッチャーオブジェクトを使用できます。

  • View -ビューは、リクエストが行われるオブジェクトです。

実装

FrontControllerDispatcherを作成して、それに応じてFrontControllerDispatcherとして機能させます。HomeViewStudentViewは、リクエストがフロントコントローラーに届く可能性のあるさまざまなビューを表します。

FrontControllerPatternDemo、デモクラスはFrontControllerを使用して、フロントコントローラーのデザインパターンを示します。

ステップ1

ビューを作成します。

HomeView.java

public class HomeView {
   public void show(){
      System.out.println("Displaying Home Page");
   }
}

StudentView.java

public class StudentView {
   public void show(){
      System.out.println("Displaying Student Page");
   }
}

ステップ2

Dispatcherを作成します。

Dispatcher.java

public class Dispatcher {
   private StudentView studentView;
   private HomeView homeView;
   public Dispatcher(){
      studentView = new StudentView();
      homeView = new HomeView();
   }

   public void dispatch(String request){
      if(request.equalsIgnoreCase("STUDENT")){
         studentView.show();
      }else{
         homeView.show();
      }	
   }
}

ステップ3

FrontControllerを作成する

Context.java

public class FrontController {
	
   private Dispatcher dispatcher;

   public FrontController(){
      dispatcher = new Dispatcher();
   }

   private boolean isAuthenticUser(){
      System.out.println("User is authenticated successfully.");
      return true;
   }

   private void trackRequest(String request){
      System.out.println("Page requested: " + request);
   }

   public void dispatchRequest(String request){
      //log each request
      trackRequest(request);
      //authenticate the user
      if(isAuthenticUser()){
         dispatcher.dispatch(request);
      }	
   }
}

ステップ4

FrontControllerを使用して、FrontControllerのデザインパターンを示します。

FrontControllerPatternDemo.java

public class FrontControllerPatternDemo {
   public static void main(String[] args) {
      FrontController frontController = new FrontController();
      frontController.dispatchRequest("HOME");
      frontController.dispatchRequest("STUDENT");
   }
}

ステップ5

出力を確認します。

Page requested: HOME
User is authenticated successfully.
Displaying Home Page
Page requested: STUDENT
User is authenticated successfully.
Displaying Student Page

インターセプトフィルターのデザインパターンは、アプリケーションの要求または応答を使用して前処理/後処理を実行する場合に使用されます。フィルタは、リクエストを実際のターゲットアプリケーションに渡す前に、リクエストに定義されて適用されます。フィルタは、認証/承認/ロギングまたはリクエストの追跡を実行してから、対応するハンドラにリクエストを渡すことができます。以下は、このタイプのデザインパターンのエンティティです。

  • Filter -リクエストハンドラによるリクエストの実行前または実行後に特定のタスクを実行するフィルタ。

  • Filter Chain -フィルターチェーンは複数のフィルターを運び、ターゲット上で定義された順序でそれらを実行するのに役立ちます。

  • Target -ターゲットオブジェクトはリクエストハンドラです

  • Filter Manager -Filter Managerは、フィルターとフィルターチェーンを管理します。

  • Client -クライアントは、ターゲットオブジェクトにリクエストを送信するオブジェクトです。

実装

エンティティを表すさまざまなオブジェクトとして、FilterChainFilterManagerTargetClientを作成します。AuthenticationFilterDebugFilterは、具象フィルターを表します。

InterceptingFilterDemoのデモクラスでは、Clientを使用してInterceptingFilterデザインパターンを示します。

ステップ1

フィルタインターフェイスを作成します。

Filter.java

public interface Filter {
   public void execute(String request);
}

ステップ2

具体的なフィルターを作成します。

AuthenticationFilter.java

public class AuthenticationFilter implements Filter {
   public void execute(String request){
      System.out.println("Authenticating request: " + request);
   }
}

DebugFilter.java

public class DebugFilter implements Filter {
   public void execute(String request){
      System.out.println("request log: " + request);
   }
}

ステップ3

ターゲットを作成する

Target.java

public class Target {
   public void execute(String request){
      System.out.println("Executing request: " + request);
   }
}

ステップ4

フィルタチェーンを作成する

FilterChain.java

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

public class FilterChain {
   private List<Filter> filters = new ArrayList<Filter>();
   private Target target;

   public void addFilter(Filter filter){
      filters.add(filter);
   }

   public void execute(String request){
      for (Filter filter : filters) {
         filter.execute(request);
      }
      target.execute(request);
   }

   public void setTarget(Target target){
      this.target = target;
   }
}

ステップ5

フィルタマネージャを作成する

FilterManager.java

public class FilterManager {
   FilterChain filterChain;

   public FilterManager(Target target){
      filterChain = new FilterChain();
      filterChain.setTarget(target);
   }
   public void setFilter(Filter filter){
      filterChain.addFilter(filter);
   }

   public void filterRequest(String request){
      filterChain.execute(request);
   }
}

ステップ6

クライアントの作成

Client.java

public class Client {
   FilterManager filterManager;

   public void setFilterManager(FilterManager filterManager){
      this.filterManager = filterManager;
   }

   public void sendRequest(String request){
      filterManager.filterRequest(request);
   }
}

ステップ7

クライアントを使用して、インターセプトフィルターのデザインパターンを示します。

FrontControllerPatternDemo.java

public class InterceptingFilterDemo {
   public static void main(String[] args) {
      FilterManager filterManager = new FilterManager(new Target());
      filterManager.setFilter(new AuthenticationFilter());
      filterManager.setFilter(new DebugFilter());

      Client client = new Client();
      client.setFilterManager(filterManager);
      client.sendRequest("HOME");
   }
}

ステップ8

出力を確認します。

Authenticating request: HOME
request log: HOME
Executing request: HOME

サービスロケーターデザインパターンは、JNDIルックアップを使用してさまざまなサービスを検索する場合に使用されます。サービスのJNDIを検索するコストが高いことを考慮して、ServiceLocatorパターンはキャッシュ技術を利用します。初めてサービスが必要になると、Service LocatorはJNDIを検索し、サービスオブジェクトをキャッシュします。Service Locatorを介したさらなるルックアップまたは同じサービスがキャッシュで実行されるため、アプリケーションのパフォーマンスが大幅に向上します。以下は、このタイプのデザインパターンのエンティティです。

  • Service-リクエストを処理する実際のサービス。このようなサービスの参照は、JNDIサーバーで確認する必要があります。

  • Context / Initial Context -JNDIコンテキストは、ルックアップの目的で使用されるサービスへの参照を伝達します。

  • Service Locator --Service Locatorは、JNDIルックアップによってサービスを取得し、サービスをキャッシュするための単一の連絡先です。

  • Cache -サービスの参照を保存して再利用するためのキャッシュ

  • Client -クライアントは、ServiceLocatorを介してサービスを呼び出すオブジェクトです。

実装

エンティティを表すさまざまなオブジェクトとして、ServiceLocatorInitialContextCacheServiceを作成しますService1Service2は、具体的なサービスを表します。

ServiceLocatorPatternDemo、デモクラスはここでクライアントとして機能し、ServiceLocatorを使用してServiceLocatorデザインパターンを示します。

ステップ1

サービスインターフェイスを作成します。

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を使用して、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

オブジェクトの転送パターンは、複数の属性を持つデータをクライアントからサーバーに一度に渡す場合に使用されます。転送オブジェクトは、値オブジェクトとも呼ばれます。Transfer Objectは、getter / setterメソッドを持つ単純なPOJOクラスであり、ネットワーク経由で転送できるようにシリアル化できます。動作はありません。サーバーサイドのビジネスクラスは通常、データベースからデータをフェッチし、POJOにデータを入力してクライアントに送信するか、値で渡します。クライアントの場合、転送オブジェクトは読み取り専用です。クライアントは独自の転送オブジェクトを作成してサーバーに渡し、データベースの値を一度に更新できます。以下は、このタイプのデザインパターンのエンティティです。

  • Business Object -転送オブジェクトにデータを入力するビジネスサービス。

  • Transfer Object -属性のみを設定/取得するメソッドを持つ単純なPOJO。

  • Client -クライアントは、転送オブジェクトをビジネスオブジェクトに要求または送信します。

実装

エンティティを表すStudentBOをビジネスオブジェクトとして、Studentを転送オブジェクトとして作成します。

TransferObjectPatternDemo、デモクラスはここでクライアントとして機能し、StudentBOStudentを使用してTransfer Object DesignPatternをデモンストレーションします。

ステップ1

転送オブジェクトを作成します。

StudentVO.java

public class StudentVO {
   private String name;
   private int rollNo;

   StudentVO(String name, int rollNo){
      this.name = name;
      this.rollNo = rollNo;
   }

   public String getName() {
      return name;
   }

   public void setName(String name) {
      this.name = name;
   }

   public int getRollNo() {
      return rollNo;
   }

   public void setRollNo(int rollNo) {
      this.rollNo = rollNo;
   }
}

ステップ2

ビジネスオブジェクトを作成します。

StudentBO.java

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

public class StudentBO {
	
   //list is working as a database
   List<StudentVO> students;

   public StudentBO(){
      students = new ArrayList<StudentVO>();
      StudentVO student1 = new StudentVO("Robert",0);
      StudentVO student2 = new StudentVO("John",1);
      students.add(student1);
      students.add(student2);		
   }
   public void deleteStudent(StudentVO student) {
      students.remove(student.getRollNo());
      System.out.println("Student: Roll No " 
      + student.getRollNo() +", deleted from database");
   }

   //retrive list of students from the database
   public List<StudentVO> getAllStudents() {
      return students;
   }

   public StudentVO getStudent(int rollNo) {
      return students.get(rollNo);
   }

   public void updateStudent(StudentVO student) {
      students.get(student.getRollNo()).setName(student.getName());
      System.out.println("Student: Roll No " 
      + student.getRollNo() +", updated in the database");
   }
}

ステップ3

StudentBOを使用して、転送オブジェクトのデザインパターンを示します。

TransferObjectPatternDemo.java

public class TransferObjectPatternDemo {
   public static void main(String[] args) {
      StudentBO studentBusinessObject = new StudentBO();

      //print all students
      for (StudentVO student : studentBusinessObject.getAllStudents()) {
         System.out.println("Student: [RollNo : "
         +student.getRollNo()+", Name : "+student.getName()+" ]");
      }

      //update student
      StudentVO student =studentBusinessObject.getAllStudents().get(0);
      student.setName("Michael");
      studentBusinessObject.updateStudent(student);

      //get the student
      studentBusinessObject.getStudent(0);
      System.out.println("Student: [RollNo : "
      +student.getRollNo()+", Name : "+student.getName()+" ]");
   }
}

ステップ4

出力を確認します。

Student: [RollNo : 0, Name : Robert ]
Student: [RollNo : 1, Name : John ]
Student: Roll No 0, updated in the database
Student: [RollNo : 0, Name : Michael ]