Guide rapide des modèles de conception

Les modèles de conception représentent les meilleures pratiques utilisées par les développeurs de logiciels orientés objet expérimentés. Les modèles de conception sont des solutions aux problèmes généraux auxquels les développeurs de logiciels ont été confrontés lors du développement de logiciels. Ces solutions ont été obtenues par essais et erreurs par de nombreux développeurs de logiciels sur une période assez longue.

Qu'est-ce que Gang of Four (GOF)?

En 1994, quatre auteurs Erich Gamma, Richard Helm, Ralph Johnson et John Vlissides ont publié un livre intitulé Design Patterns - Elements of Reusable Object-Oriented Software qui a lancé le concept de Design Pattern dans le développement de logiciels.

Ces auteurs sont connus sous le nom de Gang of Four (GOF). Selon ces auteurs, les modèles de conception sont principalement basés sur les principes suivants de conception orientée objet.

  • Programme vers une interface pas une implémentation

  • Privilégier la composition d'objets par rapport à l'héritage

Utilisation du modèle de conception

Les modèles de conception ont deux utilisations principales dans le développement de logiciels.

Plateforme commune pour les développeurs

Les modèles de conception fournissent une terminologie standard et sont spécifiques à un scénario particulier. Par exemple, un modèle de conception singleton signifie l'utilisation d'un objet unique afin que tous les développeurs familiers avec un modèle de conception unique utilisent un seul objet et puissent se dire que le programme suit un modèle singleton.

Les meilleures pratiques

Les modèles de conception ont évolué sur une longue période et fournissent les meilleures solutions à certains problèmes rencontrés lors du développement de logiciels. L'apprentissage de ces modèles aide les développeurs non expérimentés à apprendre la conception de logiciels de manière simple et rapide.

Types de modèle de conception

Selon le livre de référence de modèle de conception Design Patterns - Elements of Reusable Object-Oriented Software, il existe 23 modèles de conception. Ces modèles peuvent être classés en trois catégories: modèles créatifs, structurels et comportementaux. Nous aborderons également une autre catégorie de modèles de conception: les modèles de conception J2EE.

SN Modèle et description
1 Creational Patterns
Ces modèles de conception permettent de créer des objets tout en masquant la logique de création, plutôt que d'instancier des objets directement à l'aide d'un nouvel opérateur. Cela donne au programme plus de flexibilité pour décider quels objets doivent être créés pour un cas d'utilisation donné.
2 Structural Patterns
Ces modèles de conception concernent la composition des classes et des objets. Le concept d'héritage permet de composer des interfaces et de définir des manières de composer des objets pour obtenir de nouvelles fonctionnalités.
3 Behavioral Patterns
Ces modèles de conception concernent spécifiquement la communication entre les objets.
4 J2EE Patterns
Ces modèles de conception concernent spécifiquement le niveau de présentation. Ces modèles sont identifiés par Sun Java Center.

Le modèle d'usine est l'un des modèles de conception les plus utilisés en Java. Ce type de modèle de conception fait partie du modèle de création car ce modèle fournit l'un des meilleurs moyens de créer un objet.

Dans le modèle Factory, nous créons un objet sans exposer la logique de création au client et nous référons à un objet nouvellement créé à l'aide d'une interface commune.

la mise en oeuvre

Nous allons créer une interface Shape et des classes concrètes implémentant l' interface Shape . Une classe d'usine ShapeFactory est définie comme étape suivante.

FactoryPatternDemo , notre classe de démonstration utilisera ShapeFactory pour obtenir un objet Shape . Il transmettra les informations ( CIRCLE / RECTANGLE / SQUARE ) à ShapeFactory pour obtenir le type d'objet dont il a besoin.

Étape 1

Créez une interface.

Shape.java

public interface Shape {
   void draw();
}

Étape 2

Créez des classes concrètes mettant en œuvre la même interface.

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.");
   }
}

Étape 3

Créez une fabrique pour générer un objet de classe concrète basé sur des informations données.

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;
   }
}

Étape 4

Utilisez la fabrique pour obtenir un objet de classe concrète en passant une information telle que type.

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();
   }
}

Étape 5

Vérifiez la sortie.

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

Les modèles d'usine abstraite fonctionnent autour d'une super-usine qui crée d'autres usines. Cette usine est également appelée usine d'usines. Ce type de modèle de conception fait partie du modèle de création car ce modèle fournit l'un des meilleurs moyens de créer un objet.

Dans le modèle Abstract Factory, une interface est responsable de la création d'une fabrique d'objets associés sans spécifier explicitement leurs classes. Chaque usine générée peut donner les objets selon le modèle Factory.

la mise en oeuvre

Nous allons créer une interface Shape et une classe concrète l'implémentant. Nous créons une classe de fabrique abstraite AbstractFactory comme étape suivante. La classe Factory ShapeFactory est définie, ce qui étend AbstractFactory. Une classe de créateur / générateur d'usine FactoryProducer est créée.

AbstractFactoryPatternDemo, notre classe de démonstration utilise FactoryProducer pour obtenir un objet AbstractFactory. Il transmettra des informations (CIRCLE / RECTANGLE / SQUARE pour Shape) à AbstractFactory pour obtenir le type d'objet dont il a besoin.

Étape 1

Créez une interface pour les formes.

Shape.java

public interface Shape {
   void draw();
}

Étape 2

Créez des classes concrètes mettant en œuvre la même interface.

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.");
   }
}

Étape 3

Créez une classe abstraite pour obtenir des usines pour les objets de forme normale et arrondie.

AbstractFactory.java

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

Étape 4

Créez des classes Factory étendant AbstractFactory pour générer un objet de classe concrète basé sur des informations données.

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;
   }
}

Étape 5

Créez un générateur / classe de producteur d'usine pour obtenir des usines en transmettant une information telle que Shape

FactoryProducer.java

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

Étape 6

Utilisez FactoryProducer pour obtenir AbstractFactory afin d'obtenir des fabriques de classes concrètes en passant une information telle que type.

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();
      
   }
}

Étape 7

Vérifiez la sortie.

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

Le modèle Singleton est l'un des modèles de conception les plus simples de Java. Ce type de modèle de conception relève du modèle de création car ce modèle constitue l'un des meilleurs moyens de créer un objet.

Ce modèle implique une seule classe qui est responsable de créer son propre objet tout en s'assurant qu'un seul objet est créé. Cette classe fournit un moyen d'accéder à son seul objet auquel il est possible d'accéder directement sans avoir besoin d'instancier l'objet de la classe.

la mise en oeuvre

Nous allons créer une classe SingleObject . La classe SingleObject a son constructeur privé et une instance statique d'elle-même.

La classe SingleObject fournit une méthode statique pour obtenir son instance statique vers le monde extérieur. SingletonPatternDemo , notre classe de démonstration utilisera la classe SingleObject pour obtenir un objet SingleObject .

Étape 1

Créez une classe Singleton.

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!");
   }
}

Étape 2

Obtenez le seul objet de la classe singleton.

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();
   }
}

Étape 3

Vérifiez la sortie.

Hello World!

Le modèle Builder crée un objet complexe à l'aide d'objets simples et en utilisant une approche étape par étape. Ce type de modèle de conception fait partie du modèle de création car ce modèle fournit l'un des meilleurs moyens de créer un objet.

Une classe Builder construit l'objet final étape par étape. Ce générateur est indépendant des autres objets.

la mise en oeuvre

Nous avons envisagé une analyse de rentabilisation de restauration rapide où un repas typique pourrait être un hamburger et une boisson fraîche. Burger peut être soit un Veg Burger ou Chicken Burger et sera emballé dans un emballage. La boisson froide peut être un coke ou du pepsi et sera emballée dans une bouteille.

Nous allons créer une interface Item représentant des produits alimentaires tels que des hamburgers et des boissons froides et des classes concrètes mettant en œuvre l' interface Item et une interface Packing représentant l'emballage des produits alimentaires et des classes concrètes implémentant l' interface Packing car le hamburger serait emballé dans un emballage et froid la boisson serait emballée sous forme de bouteille.

Nous créons ensuite une classe Meal ayant ArrayList of Item et un MealBuilder pour construire différents types d' objets Meal en combinant Item . BuilderPatternDemo , notre classe de démonstration utilisera MealBuilder pour créer un repas .

Étape 1

Créez un élément d'interface représentant un aliment et un emballage.

Item.java

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

Packing.java

public interface Packing {
   public String pack();
}

Étape 2

Créez des classes concrètes implémentant l'interface Packing.

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";
   }
}

Étape 3

Créez des classes abstraites mettant en œuvre l'interface d'élément fournissant des fonctionnalités par défaut.

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();
}

Étape 4

Créer des classes concrètes étendant les classes Burger et 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";
   }
}

Étape 5

Créez une classe Meal avec les objets Item définis ci-dessus.

Repas.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());
      }		
   }	
}

Étape 6

Créez une classe MealBuilder, la classe de générateur réelle chargée de créer des objets 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;
   }
}

Étape 7

BuiderPatternDemo utilise MealBuider pour illustrer le modèle de générateur.

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());
   }
}

Étape 8

Vérifiez la sortie.

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

Le modèle de prototype fait référence à la création d'un objet dupliqué tout en gardant à l'esprit les performances. Ce type de modèle de conception relève du modèle de création car ce modèle constitue l'un des meilleurs moyens de créer un objet.

Ce modèle implique l'implémentation d'une interface prototype qui dit de créer un clone de l'objet courant. Ce modèle est utilisé lorsque la création directe d'objet est coûteuse. Par exemple, un objet doit être créé après une opération de base de données coûteuse. Nous pouvons mettre en cache l'objet, retourner son clone à la prochaine demande et mettre à jour la base de données au fur et à mesure des besoins, réduisant ainsi les appels à la base de données.

la mise en oeuvre

Nous allons créer une classe abstraite Shape et des classes concrètes étendant la classe Shape . Une classe ShapeCache est définie comme une forme étape suivante qui stocke des objets dans un Hashtable et retourne leur clone sur demande.

PrototypPatternDemo , notre classe de démonstration utilisera la classe ShapeCache pour obtenir un objet Shape .

Étape 1

Créez une classe abstraite implémentant une interface 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;
   }
}

Étape 2

Créez des classes concrètes étendant la classe ci-dessus.

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.");
   }
}

Étape 3

Créer une classe pour obtenir des classes ConCreate la base de données et de les stocker dans un Hashtable .

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);
   }
}

Étape 4

PrototypePatternDemo utilise la classe ShapeCache pour obtenir des clones de formes stockées dans un Hashtable .

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());		
   }
}

Étape 5

Vérifiez la sortie.

Shape : Circle
Shape : Square
Shape : Rectangle

Le modèle d'adaptateur fonctionne comme un pont entre deux interfaces incompatibles. Ce type de modèle de conception relève d'un modèle structurel car ce modèle combine la capacité de deux interfaces indépendantes.

Ce modèle implique une seule classe qui est chargée de joindre les fonctionnalités d'interfaces indépendantes ou incompatibles. Un exemple concret pourrait être un cas de lecteur de carte qui agit comme un adaptateur entre la carte mémoire et un ordinateur portable. Vous branchez la carte mémoire dans le lecteur de carte et le lecteur de carte dans l'ordinateur portable afin que la carte mémoire puisse être lue via un ordinateur portable.

Nous démontrons l'utilisation du modèle d'adaptateur via l'exemple suivant dans lequel un lecteur audio ne peut lire que des fichiers mp3 et souhaite utiliser un lecteur audio avancé capable de lire des fichiers vlc et mp4.

la mise en oeuvre

Nous avons une interface MediaPlayer interface et une classe de béton AudioPlayer mise en œuvre du MediaPlayer interface. AudioPlayer peut lire des fichiers audio au format mp3 par défaut.

Nous avons une autre interface AdvancedMediaPlayer et des classes concrètes implémentant l' interface AdvancedMediaPlayer. Ces classes peuvent lire des fichiers au format vlc et mp4.

Nous voulons que AudioPlayer puisse également lire d'autres formats. Pour y parvenir, nous avons créé une classe adaptateur MediaAdapter qui implémente l' MediaPlayer interface et utilisations AdvancedMediaPlayer objets pour lire le format requis.

AudioPlayer utilise la classe d'adaptateur MediaAdapter en lui passant le type audio souhaité sans connaître la classe réelle qui peut lire le format souhaité. AdapterPatternDemo , notre classe de démonstration utilisera la classe AudioPlayer pour lire différents formats.

Étape 1

Créez des interfaces pour Media Player et Advanced Media Player.

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);
}

Étape 2

Créez des classes concrètes mettant en œuvre l' interface 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);		
   }
}

Étape 3

Créez une classe d'adaptateur implémentant l' interface 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);
      }
   }
}

Étape 4

Créez une classe concrète implémentant l' interface 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");
      }
   }   
}

Étape 5

Utilisez AudioPlayer pour lire différents types de formats audio.

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");
   }
}

Étape 6

Vérifiez la sortie.

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 est utilisé lorsque nous devons découpler une abstraction de son implémentation afin que les deux puissent varier indépendamment. Ce type de modèle de conception relève d'un modèle structurel car ce modèle dissocie la classe d'implémentation et la classe abstraite en fournissant une structure de pont entre elles.

Ce modèle implique une interface qui agit comme un pont qui rend la fonctionnalité des classes concrètes indépendante des classes d'implémentation d'interface. Les deux types de classes peuvent être modifiés structurellement sans s’affecter mutuellement.

Nous démontrons l'utilisation du modèle Bridge via l'exemple suivant dans lequel un cercle peut être dessiné dans différentes couleurs en utilisant la même méthode de classe abstraite mais différentes classes d'implémentation de pont.

la mise en oeuvre

Nous avons une interface DrawAPI qui agit comme un implémenteur de pont et des classes concrètes RedCircle , GreenCircle implémentant l' interface DrawAPI . Shape est une classe abstraite et utilisera l'objet de DrawAPI . BridgePatternDemo , notre classe de démonstration utilisera la classe Shape pour dessiner des cercles de couleurs différentes.

Étape 1

Créer une interface de mise en œuvre de pont.

DrawAPI.java

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

Étape 2

Créez des classes d'implémentation de ponts concrets implémentant l' interface 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 +"]");
   }
}

Étape 3

Créez une classe abstraite Shape à l'aide de l' interface DrawAPI .

Shape.java

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

Étape 4

Créez une classe concrète implémentant l' interface 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);
   }
}

Étape 5

Utilisez les classes Shape et DrawAPI pour dessiner différents cercles colorés.

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();
   }
}

Étape 6

Vérifiez la sortie.

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

Le modèle de filtre ou modèle de critère est un modèle de conception qui permet aux développeurs de filtrer un ensemble d'objets, en utilisant différents critères, en les chaînant de manière découplée à travers des opérations logiques. Ce type de modèle de conception relève du modèle structurel car ce modèle combine plusieurs critères pour obtenir un seul critère.

la mise en oeuvre

Nous allons créer un objet Person , une interface Criteria et des classes concrètes implémentant cette interface pour filtrer la liste des objets Person . CriteriaPatternDemo , notre classe de démonstration utilise des objets Criteria pour filtrer les objets List of Person en fonction de divers critères et de leurs combinaisons.

Étape 1

Créez une classe sur laquelle les critères doivent être appliqués.

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;
   }	
}

Étape 2

Créez une interface pour les critères.

Criteria.java

import java.util.List;

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

Étape 3

Créez des classes concrètes mettant en œuvre l' interface 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;
   }
}

Étape 4

Utilisez différents critères et leur combinaison pour filtrer les personnes.

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()
            +" ]");
      }
   }      
}

Étape 5

Vérifiez la sortie.

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 ]

Le motif composite est utilisé lorsque nous devons traiter un groupe d'objets de la même manière comme un seul objet. Le motif composite compose des objets en termes de structure arborescente pour représenter une partie ainsi que toute la hiérarchie. Ce type de modèle de conception relève d'un modèle structurel car ce modèle crée une structure arborescente de groupe d'objets.

Ce modèle crée une classe contenant un groupe de ses propres objets. Cette classe fournit des moyens de modifier son groupe de mêmes objets.

Nous démontrons l'utilisation du modèle composite via l'exemple suivant dans lequel montrer la hiérarchie des employés d'une organisation.

la mise en oeuvre

Nous avons une classe Employee qui agit comme une classe d'acteur de modèle composite. CompositePatternDemo , notre classe de démonstration utilisera la classe Employee pour ajouter une hiérarchie au niveau du service et imprimer tous les employés.

Étape 1

Créez une classe Employee ayant une liste d' objets 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+" ]");
   }   
}

Étape 2

Utilisez la classe Employee pour créer et imprimer la hiérarchie des employés.

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);
         }
      }		
   }
}

Étape 3

Vérifiez la sortie.

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 ]

Le motif Decorator permet d'ajouter de nouvelles fonctionnalités à un objet existant sans altérer sa structure. Ce type de modèle de conception relève d'un modèle structurel car ce modèle agit comme un wrapper pour la classe existante.

Ce modèle crée une classe de décorateur qui encapsule la classe d'origine et fournit des fonctionnalités supplémentaires en préservant la signature des méthodes de classe.

Nous démontrons l'utilisation du motif Decorator via l'exemple suivant dans lequel nous décorerons une forme avec une certaine couleur sans modifier la classe de forme.

la mise en oeuvre

Nous allons créer une interface Shape et des classes concrètes implémentant l' interface Shape . Nous créons ensuite une classe de décorateur abstraite ShapeDecorator implémentant l' interface Shape et ayant l' objet Shape comme variable d'instance.

RedShapeDecorator est une classe concrète implémentant ShapeDecorator .

DecoratorPatternDemo , notre classe de démonstration utilisera RedShapeDecorator pour décorer les objets Shape .

Étape 1

Créez une interface.

Shape.java

public interface Shape {
   void draw();
}

Étape 2

Créez des classes concrètes mettant en œuvre la même interface.

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");
   }
}

Étape 3

Créez une classe de décorateur abstraite implémentant l' interface Shape .

ShapeDecorator.java

public abstract class ShapeDecorator implements Shape {
   protected Shape decoratedShape;

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

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

Étape 4

Créez une classe de décorateur en béton étendant la classe 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");
   }
}

Étape 5

Utilisez RedShapeDecorator pour décorer les objets Shape .

DécorateurPatternDemo.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();
   }
}

Étape 6

Vérifiez la sortie.

Circle with normal border
Shape: Circle

Circle of red border
Shape: Circle
Border Color: Red

Rectangle of red border
Shape: Rectangle
Border Color: Red

Le modèle de façade cache les complexités du système et fournit une interface au client à l'aide de laquelle le client peut accéder au système. Ce type de modèle de conception relève d'un modèle structurel car ce modèle ajoute une interface au système sortant pour masquer ses complexités.

Ce modèle implique une classe unique qui fournit des méthodes simplifiées requises par le client et délègue les appels aux méthodes de classes système existantes.

la mise en oeuvre

Nous allons créer une interface Shape et des classes concrètes implémentant l' interface Shape . Une classe de façade ShapeMaker est définie comme étape suivante.

La classe ShapeMaker utilise les classes concrètes pour déléguer les appels d'utilisateurs à ces classes. FacadePatternDemo , notre classe de démonstration utilisera la classe ShapeMaker pour afficher les résultats.

Étape 1

Créez une interface.

Shape.java

public interface Shape {
   void draw();
}

Étape 2

Créez des classes concrètes mettant en œuvre la même interface.

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()");
   }
}

Étape 3

Créez une classe de façade.

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();
   }
}

Étape 4

Utilisez la façade pour dessiner différents types de formes.

FaçadePatternDemo.java

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

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

Étape 5

Vérifiez la sortie.

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

Le modèle Flyweight est principalement utilisé pour réduire le nombre d'objets créés, pour réduire l'empreinte mémoire et augmenter les performances. Ce type de modèle de conception relève d'un modèle structurel car ce modèle fournit des moyens de réduire le nombre d'objets, améliorant ainsi la structure des objets requis par l'application.

Le modèle Flyweight tente de réutiliser des objets de type similaire déjà existants en les stockant et en crée un nouvel objet lorsqu'aucun objet correspondant n'est trouvé. Nous allons démontrer ce modèle en dessinant 20 cercles d'emplacements différents, mais nous ne créerons que 5 objets. Seules 5 couleurs sont disponibles, donc la propriété de couleur est utilisée pour vérifier les objets Circle déjà existants .

la mise en oeuvre

Nous allons créer une interface Shape et une classe concrète Circle implémentant l' interface Shape . Une classe d'usine ShapeFactory est définie comme étape suivante.

ShapeFactory a un HashMap de Circle ayant la clé comme couleur de l' objet Circle . Chaque fois qu'une demande vient de créer un cercle de couleur particulière à ShapeFactory . ShapeFactory vérifie l'objet Circle dans son HashMap , si l'objet de Circle est trouvé, cet objet est retourné sinon un nouvel objet est créé, stocké dans hashmap pour une utilisation future et renvoyé au client.

FlyWeightPatternDemo , notre classe de démonstration utilisera ShapeFactory pour obtenir un objet Shape . Il transmettra des informations ( rouge / vert / bleu / noir / blanc ) à ShapeFactory pour obtenir le cercle de la couleur souhaitée dont il a besoin.

Étape 1

Créez une interface.

Shape.java

public interface Shape {
   void draw();
}

Étape 2

Créez une classe concrète implémentant la même interface.

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);
   }
}

Étape 3

Créez une fabrique pour générer un objet de classe concrète basé sur des informations données.

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;
   }
}

Étape 4

Utilisez la fabrique pour obtenir un objet de classe concrète en passant une information telle que la couleur.

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);
   }
}

Étape 5

Vérifiez la sortie.

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

Dans le modèle Proxy, une classe représente la fonctionnalité d'une autre classe. Ce type de modèle de conception relève du modèle structurel.

Dans le modèle Proxy, nous créons un objet ayant un objet original pour interfacer ses fonctionnalités avec le monde extérieur.

la mise en oeuvre

Nous allons créer une interface Image et des classes concrètes implémentant l' interface Image . ProxyImage est une classe proxy pour réduire l'empreinte mémoire du chargement des objets RealImage .

ProxyPatternDemo , notre classe de démonstration utilisera ProxyImage pour obtenir un objet Image à charger et à afficher selon ses besoins.

Étape 1

Créez une interface.

Image.java

public interface Image {
   void display();
}

Étape 2

Créez des classes concrètes mettant en œuvre la même interface.

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();
   }
}

Étape 3

Utilisez ProxyImage pour obtenir l'objet de la classe RealImage si nécessaire.

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(); 	
   }
}

Étape 4

Vérifiez la sortie.

Loading test_10mb.jpg
Displaying test_10mb.jpg

Displaying test_10mb.jpg

Comme son nom l'indique, le modèle de chaîne de responsabilité crée une chaîne d'objets récepteurs pour une demande. Ce modèle découple l'expéditeur et le destinataire d'une demande en fonction du type de demande. Ce modèle relève de modèles comportementaux.

Dans ce modèle, chaque récepteur contient normalement une référence à un autre récepteur. Si un objet ne peut pas gérer la demande, il la transmet au récepteur suivant et ainsi de suite.

la mise en oeuvre

Nous avons créé une classe abstraite AbstractLogger avec un niveau de journalisation. Ensuite, nous avons créé trois types de loggers étendant le AbstractLogger . Chaque enregistreur vérifie le niveau du message à son niveau et imprime en conséquence sinon n'imprime pas et transmet le message à son enregistreur suivant.

Étape 1

Créez une classe de journalisation abstraite.

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);
	
}

Étape 2

Créez des classes concrètes étendant le logger.

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);
   }
}

Étape 3

Créez différents types d'enregistreurs. Attribuez-leur des niveaux d'erreur et définissez le prochain enregistreur dans chaque enregistreur. L'enregistreur suivant dans chaque enregistreur représente la partie de la chaîne.

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.");
   }
}

Étape 4

Vérifiez la sortie.

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.

Le modèle de commande est un modèle de conception basé sur les données et relève de la catégorie de modèle comportemental. Une demande est encapsulée sous un objet en tant que commande et transmise à l'objet appelant. L'objet invocateur recherche l'objet approprié qui peut gérer cette commande et passer la commande à l'objet correspondant et cet objet exécute la commande.

la mise en oeuvre

Nous avons créé une interface Order qui agit comme une commande. Nous avons créé une classe Stock qui agit comme une requête. Nous avons des classes de commandes concrètes BuyStock et SellStock mise en œuvre ordre interface qui fera le traitement de commande réelle. Un courtier de classe est créé qui agit comme un objet invocateur. Il peut prendre commande et passer des commandes.

L' objet Broker utilise un modèle de commande pour identifier quel objet exécutera quelle commande en fonction du type de commande. CommandPatternDemo , notre classe de démonstration utilisera la classe Broker pour démontrer le modèle de commande.

Étape 1

Créez une interface de commande.

Order.java

public interface Order {
   void execute();
}

Étape 2

Créez une classe de requête.

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");
   }
}

Étape 3

Créez des classes concrètes implémentant l' interface 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();
   }
}

Étape 4

Créez une classe d'invocateur de commande.

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();
   }
}

Étape 5

Utilisez la classe Broker pour prendre et exécuter des commandes.

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();
   }
}

Étape 6

Vérifiez la sortie.

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

Le modèle d'interprétation permet d'évaluer la grammaire ou l'expression de la langue. Ce type de modèle relève de modèles de comportement. Ce modèle implique l'implémentation d'une interface d'expression qui dit d'interpréter un contexte particulier. Ce modèle est utilisé dans l'analyse SQL, le moteur de traitement de symboles, etc.

la mise en oeuvre

Nous allons créer une interface Expression et des classes concrètes implémentant l' interface Expression . Une classe TerminalExpression est définie qui agit comme un interprète principal du contexte en question. D'autres classes OrExpression , AndExpression sont utilisées pour créer des expressions combinatoires.

InterpreterPatternDemo , notre classe de démonstration utilisera la classe Expression pour créer des règles et démontrer l'analyse des expressions.

Étape 1

Créez une interface d'expression.

Expression.java

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

Étape 2

Créez des classes concrètes mettant en œuvre l'interface ci-dessus.

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);
   }
}

Étape 3

InterpreterPatternDemo utilise la classe Expression pour créer des règles, puis les analyser.

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"));
   }
}

Étape 4

Vérifiez la sortie.

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

Le modèle Iterator est un modèle de conception très couramment utilisé dans l'environnement de programmation Java et .Net. Ce modèle est utilisé pour obtenir un moyen d'accéder aux éléments d'un objet de collection de manière séquentielle sans avoir besoin de connaître sa représentation sous-jacente.

Le modèle d'itérateur relève de la catégorie de modèle comportemental.

la mise en oeuvre

Nous allons créer une interface Iterator qui raconte la méthode de navigation et une interface Container qui retransmet l'itérateur. Les classes concrètes implémentant l' interface Container seront chargées d'implémenter l' interface Iterator et de l'utiliser

IteratorPatternDemo , notre classe de démonstration utilisera NamesRepository , une implémentation de classe concrète pour imprimer un Names stocké en tant que collection dans NamesRepository .

Étape 1

Créez des interfaces.

Iterator.java

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

Container.java

public interface Container {
   public Iterator getIterator();
}

Étape 2

Créez une classe concrète implémentant l' interface Container . Cette classe a la classe interne NameIterator implémentant l' interface Iterator .

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;
      }		
   }
}

Étape 3

Utilisez le NameRepository pour obtenir les noms d'itérateur et d'impression.

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);
      } 	
   }
}

Étape 4

Vérifiez la sortie.

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

Le modèle Mediator est utilisé pour réduire la complexité de la communication entre plusieurs objets ou classes. Ce modèle fournit une classe de médiateur qui gère normalement toutes les communications entre les différentes classes et prend en charge la facilité de maintenance du code par couplage lâche. Le modèle de médiateur relève de la catégorie de modèle de comportement.

la mise en oeuvre

Nous présentons un modèle de médiateur par exemple d'une salle de conversation où plusieurs utilisateurs peuvent envoyer des messages à la salle de conversation et il est de la responsabilité de la salle de conversation de montrer les messages à tous les utilisateurs. Nous avons créé deux classes ChatRoom et User . Les objets utilisateur utiliseront la méthode ChatRoom pour partager leurs messages.

MediatorPatternDemo , notre classe de démonstration utilisera des objets User pour montrer la communication entre eux.

Étape 1

Créez une classe de médiateur.

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);
   }
}

Étape 2

Créer une classe d'utilisateurs

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);
   }
}

Étape 3

Utilisez l' objet Utilisateur pour afficher les communications entre eux.

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!");
   }
}

Étape 4

Vérifiez la sortie.

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

Le modèle Memento est utilisé pour réduire l'endroit où nous voulons restaurer l'état d'un objet à un état antérieur. Le modèle Memento appartient à la catégorie des modèles comportementaux.

la mise en oeuvre

Le modèle Memento utilise trois classes d'acteurs. Memento contient l'état d'un objet à restaurer. Originator crée et stocke les états dans les objets Memento et dans l'objet Caretaker qui est chargé de restaurer l'état de l'objet à partir de Memento. Nous avons créé les classes Memento , Originator et CareTaker .

MementoPatternDemo , notre classe de démonstration utilisera les objets CareTaker et Originator pour montrer la restauration des états des objets.

Étape 1

Créez une classe Memento.

Memento.java

public class Memento {
   private String state;

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

   public String getState(){
      return state;
   }	
}

Étape 2

Créer une classe Originator

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();
   }
}

Étape 3

Créer une classe 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);
   }
}

Étape 4

Utilisez les objets CareTaker et 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());
   }
}

Étape 5

Vérifiez la sortie.

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

Le modèle d'observateur est utilisé lorsqu'il existe une relation un à plusieurs entre les objets, par exemple si un objet est modifié, ses objets dépendants doivent être notifiés automatiquement. Le modèle d'observateur relève de la catégorie de modèle comportemental.

la mise en oeuvre

Le modèle d'observateur utilise trois classes d'acteurs. Sujet, observateur et client. Subject, un objet ayant des méthodes pour attacher et désattacher des observateurs à un objet client. Nous avons créé des classes Subject , une classe abstraite Observer et des classes concrètes étendant la classe abstraite Observer .

ObserverPatternDemo , notre classe de démonstration utilisera des objets de classe Subject et concret pour montrer le modèle d'observateur en action.

Étape 1

Créez une classe de sujet.

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();
      }
   } 	
}

Étape 2

Créez une classe d'observateur.

Observer.java

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

Étape 3

Créer des classes d'observateurs concrètes

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() ); 
   }
}

Étape 4

Utilisez des objets d'observateur Sujet et concret.

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);
   }
}

Étape 5

Vérifiez la sortie.

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

Dans le modèle d'état, le comportement d'une classe change en fonction de son état. Ce type de modèle de conception relève du modèle de comportement.

Dans State pattern, nous créons des objets qui représentent divers états et un objet de contexte dont le comportement varie à mesure que son objet d'état change.

la mise en oeuvre

Nous allons créer une interface State définissant une action et des classes d'état concrètes implémentant l' interface State . Le contexte est une classe qui porte un état.

StaePatternDemo , notre classe de démonstration utilisera des objets de contexte et d'état pour démontrer le changement de comportement de contexte en fonction du type d'état dans lequel il se trouve.

Étape 1

Créez une interface.

Image.java

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

Étape 2

Créez des classes concrètes mettant en œuvre la même interface.

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";
   }
}

Étape 3

Créer une classe de contexte .

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;
   }
}

Étape 4

Utilisez le contexte pour voir le changement de comportement lorsque l' état change.

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());
   }
}

Étape 5

Vérifiez la sortie.

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

Dans le modèle d'objet nul, un objet nul remplace la vérification de l'instance d'objet NULL. Au lieu de mettre if check pour une valeur nulle, Null Object reflète une relation ne rien faire. Un tel objet Null peut également être utilisé pour fournir un comportement par défaut dans le cas où les données ne sont pas disponibles.

Dans le modèle Null Object, nous créons une classe abstraite spécifiant les différentes opérations à effectuer, des classes concrètes étendant cette classe et une classe d'objet nulle fournissant une implémentation ne rien faire de cette classe et sera utilisée sans problème là où nous devons vérifier la valeur nulle.

la mise en oeuvre

Nous allons créer une classe abstraite AbstractCustomer définissant les opearations, ici le nom du client et des classes concrètes étendant la classe AbstractCustomer . Une classe d'usine CustomerFactory est créée pour renvoyer des objets RealCustomer ou NullCustomer en fonction du nom du client qui lui a été transmis.

NullPatternDemo , notre classe de démonstration utilisera CustomerFactory pour démontrer l'utilisation du modèle Null Object.

Étape 1

Créez une classe abstraite.

AbstractCustomer.java

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

Étape 2

Créez des classes concrètes étendant la classe ci-dessus.

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;
   }
}

Étape 3

Créez la classe 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();
   }
}

Étape 4

Utilisez CustomerFactory pour obtenir des objets RealCustomer ou NullCustomer en fonction du nom du client qui lui a été transmis.

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());
   }
}

Étape 5

Vérifiez la sortie.

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

Dans le modèle de stratégie, un comportement de classe ou son algorithme peut être modifié au moment de l'exécution. Ce type de modèle de conception relève du modèle de comportement.

Dans le modèle de stratégie, nous créons des objets qui représentent diverses stratégies et un objet de contexte dont le comportement varie selon son objet de stratégie. L'objet de stratégie modifie l'algorithme d'exécution de l'objet de contexte.

la mise en oeuvre

Nous allons créer une interface de stratégie définissant une action et des classes de stratégie concrètes implémentant l' interface de stratégie . Le contexte est une classe qui utilise une stratégie.

StrategyPatternDemo , notre classe de démonstration utilisera des objets de contexte et de stratégie pour démontrer le changement de comportement de contexte en fonction de la stratégie qu'elle déploie ou utilise.

Étape 1

Créez une interface.

Strategy.java

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

Étape 2

Créez des classes concrètes mettant en œuvre la même interface.

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;
   }
}

Étape 3

Créer une classe de contexte .

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);
   }
}

Étape 4

Utilisez le contexte pour voir le changement de comportement lorsqu'il modifie sa stratégie .

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));
   }
}

Étape 5

Vérifiez la sortie.

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

Dans le modèle de modèle, une classe abstraite expose les moyens / modèles définis pour exécuter ses méthodes. Ses sous-classes peuvent remplacer les implémentations de méthode selon les besoins, mais l'appel doit être de la même manière que celle définie par une classe abstraite. Ce modèle appartient à la catégorie des modèles de comportement.

la mise en oeuvre

Nous allons créer une classe abstraite Game définissant les opérations avec une méthode de modèle définie pour être finale afin qu'elle ne puisse pas être remplacée. Le cricket et le football sont des classes concrètes qui étendent le jeu et écrasent ses méthodes.

TemplatePatternDemo , notre classe de démonstration utilisera Game pour démontrer l'utilisation du modèle de modèle.

Étape 1

Créez une classe abstraite avec une méthode de modèle finale.

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();
   }
}

Étape 2

Créez des classes concrètes étendant la classe ci-dessus.

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!");
   }
}

Étape 3

Utilisez le jeu méthode modèle de jeu () pour démontrer d' une manière définie de jouer jeu.

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();		
   }
}

Étape 4

Vérifiez la sortie.

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!

Dans le modèle de visiteur, nous utilisons une classe de visiteur qui change l'algorithme d'exécution d'une classe d'élément. De cette manière, l'algorithme d'exécution de l'élément peut varier à mesure que le visiteur varie. Ce modèle appartient à la catégorie des modèles de comportement. Selon le modèle, l'objet élément doit accepter l'objet visiteur afin que l'objet visiteur gère l'opération sur l'objet élément.

la mise en oeuvre

Nous allons créer une interface ComputerPart définissant l'opearation d'acceptation. Keyboard , Mouse , Monitor et Computer sont des classes concrètes mettant en œuvre l' interface ComputerPart . Nous définirons une autre interface ComputerPartVisitor qui définira les opérations d'une classe de visiteur. L'ordinateur utilise un visiteur concret pour effectuer l'action correspondante.

VisitorPatternDemo , notre classe de démonstration utilisera les classes Computer , ComputerPartVisitor pour démontrer l'utilisation du modèle de visiteur.

Étape 1

Définissez une interface pour représenter l'élément.

ComputerPart.java

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

Étape 2

Créez des classes concrètes étendant la classe ci-dessus.

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);
   }
}

Étape 3

Définissez une interface pour représenter le visiteur.

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);
}

Étape 4

Créez un visiteur concret mettant en œuvre la classe ci-dessus.

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.");
   }
}

Étape 5

Utilisez ComputerPartDisplayVisitor pour afficher des parties de l' ordinateur .

VisitorPatternDemo.java

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

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

Étape 6

Vérifiez la sortie.

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

MVC Pattern signifie Model-View-Controller Pattern. Ce modèle est utilisé pour séparer les préoccupations de l'application.

  • Model- Le modèle représente un objet ou JAVA POJO transportant des données. Il peut également avoir une logique pour mettre à jour le contrôleur si ses données changent.

  • View - La vue représente la visualisation des données que contient le modèle.

  • Controller- Le contrôleur agit à la fois sur le modèle et sur la vue. Il contrôle le flux de données dans l'objet modèle et met à jour la vue chaque fois que les données changent. Il maintient la vue et le modèle séparés.

la mise en oeuvre

Nous allons créer un objet Student faisant office de modèle. StudentView sera une classe de vue qui peut imprimer les détails de l'étudiant sur la console et StudentController est la classe de contrôleur chargée de stocker les données dans l' objet Student et de mettre à jour la vue StudentView en conséquence.

MVCPatternDemo , notre classe de démonstration utilisera StudentController pour démontrer l'utilisation du modèle MVC.

Étape 1

Créer un modèle.

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;
   }
}

Étape 2

Créer une vue.

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);
   }
}

Étape 3

Créer un contrôleur.

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());
   }	
}

Étape 4

Utilisez les méthodes StudentController pour illustrer l'utilisation du modèle de conception 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;
   }
}

Étape 5

Vérifiez la sortie.

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

Business Delegate Pattern est utilisé pour découpler le niveau de présentation et le niveau métier. Il est essentiellement utilisé pour réduire la fonctionnalité de communication ou de recherche à distance au code de niveau entreprise dans le code de niveau de présentation. Au niveau commercial, nous suivons les entités.

  • Client - Le code de niveau de présentation peut être un code JSP, un servlet ou une interface utilisateur.

  • Business Delegate - Une classe de point d'entrée unique pour les entités clientes afin de fournir un accès aux méthodes Business Service.

  • LookUp Service - L'objet de service de recherche est chargé d'obtenir l'implémentation métier relative et de fournir l'accès aux objets métier à l'objet délégué métier.

  • Business Service- Interface de service aux entreprises. Les classes concrètes implémentent ce service métier pour fournir une logique d'implémentation métier réelle.

la mise en oeuvre

Nous allons créer un Client , BusinessDelegate , BusinessService , LookUpService , JMSService et EJBService représentant diverses entités du modèle Business Delegate.

BusinessDelegatePatternDemo , notre classe de démonstration utilisera BusinessDelegate et Client pour démontrer l'utilisation du modèle Business Delegate.

Étape 1

Créez une interface BusinessService.

BusinessService.java

public interface BusinessService {
   public void doProcessing();
}

Étape 2

Créez des classes de service 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");
   }
}

Étape 3

Créez un service de recherche d'entreprise.

BusinessLookUp.java

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

Étape 4

Créer un délégué commercial.

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();		
   }
}

Étape 5

Créer un client.

Student.java

public class Client {
	
   BusinessDelegate businessService;

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

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

Étape 6

Utilisez les classes BusinessDelegate et Client pour illustrer le modèle de délégué commercial.

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();
   }
}

Étape 7

Vérifiez la sortie.

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

Le modèle d'entité composite est utilisé dans le mécanisme de persistance EJB. Une entité composite est un bean entité EJB qui représente un graphe d'objets. Lorsqu'une entité composite est mise à jour, les beans objets dépendants en interne sont mis à jour automatiquement comme étant gérés par le bean entité EJB. Voici les participants à Composite Entity Bean.

  • Composite Entity - Il s'agit d'un bean entité primaire.Il peut être à grain grossier ou peut contenir un objet à grain grossier à utiliser à des fins de persistance.

  • Coarse-Grained Object-Cet objet contient des objets dépendants. Il a son propre cycle de vie et gère également le cycle de vie des objets dépendants.

  • Dependent Object - Les objets dépendants sont un objet qui dépend d'un objet à grain grossier pour son cycle de vie de persistance.

  • Strategies - Stratégies représente comment mettre en œuvre une entité composite.

la mise en oeuvre

Nous allons créer un objet CompositeEntity agissant en tant que CompositeEntity. CoarseGrainedObject sera une classe qui contient des objets dépendants. CompositeEntityPatternDemo , notre classe de démonstration utilisera la classe Client pour démontrer l'utilisation du modèle d'entité composite.

Étape 1

Créez des objets dépendants.

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;
   }
}

Étape 2

Créer un objet à gros grains.

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()};
   }
}

Étape 3

Créer une entité composite.

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();
   }
}

Étape 4

Créez la classe Client pour utiliser l'entité composite.

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);
   }
}

Étape 5

Utilisez le client pour démontrer l'utilisation du modèle de conception d'entité composite.

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();
   }
}

Étape 6

Vérifiez la sortie.

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

Le modèle d'objet d'accès aux données ou le modèle DAO est utilisé pour séparer les données de bas niveau accédant à l'API ou aux opérations des services métier de haut niveau. Voici les participants au modèle d'objet d'accès aux données.

  • Data Access Object Interface - Cette interface définit les opérations standard à effectuer sur un (des) objet (s) modèle (s).

  • Data Access Object concrete class-Cette classe implémente l'interface ci-dessus. Cette classe est chargée d'obtenir les données d'une source de données qui peut être database / xml ou tout autre mécanisme de stockage.

  • Model Object or Value Object - Cet objet est un simple POJO contenant des méthodes get / set pour stocker les données récupérées à l'aide de la classe DAO.

la mise en oeuvre

Nous allons créer un objet Student agissant comme un objet modèle ou valeur. StudentDao est l'interface d'objet d'accès aux données. StudentDaoImpl est une classe concrète implémentant l'interface d'objet d'accès aux données. DaoPatternDemo , notre classe de démonstration utilisera StudentDao pour démontrer l'utilisation du modèle d'objet d'accès aux données.

Étape 1

Créer un objet de valeur.

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;
   }
}

Étape 2

Créer une interface d'objet d'accès aux données.

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);
}

Étape 3

Créez une classe concrète implémentant l'interface ci-dessus.

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");
   }
}

Étape 4

Utilisez StudentDao pour illustrer l'utilisation du modèle d'objet d'accès aux données.

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()+" ]");		
   }
}

Étape 5

Vérifiez la sortie.

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

Le modèle de conception de contrôleur frontal est utilisé pour fournir un mécanisme de gestion des demandes centralisé afin que toutes les demandes soient traitées par un seul gestionnaire. Ce gestionnaire peut effectuer l'authentification / l'autorisation / la journalisation ou le suivi de la demande, puis transmettre les demandes aux gestionnaires correspondants. Voici les entités de ce type de modèle de conception.

  • Front Controller - Un seul gestionnaire pour tout type de demande venant de l'application (soit basée sur le Web / basée sur le bureau).

  • Dispatcher - Le contrôleur frontal peut utiliser un objet dispatcher qui peut envoyer la demande au gestionnaire spécifique correspondant.

  • View - Les vues sont l'objet pour lequel les demandes sont faites.

la mise en oeuvre

Nous allons créer un FrontController , Dispatcher pour agir en tant que Front Controller et Dispatcher en conséquence. HomeView et StudentView représentent diverses vues pour lesquelles les demandes peuvent arriver au contrôleur frontal.

FrontControllerPatternDemo , notre classe de démonstration utilisera FrontController pour démontrer le modèle de conception de contrôleur frontal.

Étape 1

Créez des vues.

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");
   }
}

Étape 2

Créez 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();
      }	
   }
}

Étape 3

Créer 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);
      }	
   }
}

Étape 4

Utilisez FrontController pour illustrer le modèle de conception de contrôleur frontal.

FrontControllerPatternDemo.java

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

Étape 5

Vérifiez la sortie.

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

Le modèle de conception de filtre d'interception est utilisé lorsque nous voulons effectuer un pré-traitement / post-traitement avec une demande ou une réponse de l'application. Les filtres sont définis et appliqués sur la demande avant de transmettre la demande à l'application cible réelle. Les filtres peuvent effectuer l'authentification / l'autorisation / la journalisation ou le suivi de la demande, puis transmettre les demandes aux gestionnaires correspondants. Voici les entités de ce type de modèle de conception.

  • Filter - Filtre qui effectuera certaines tâches avant ou après l'exécution de la requête par le gestionnaire de requête.

  • Filter Chain - La chaîne de filtres comporte plusieurs filtres et aide à les exécuter dans l'ordre défini sur la cible.

  • Target - L'objet cible est le gestionnaire de requêtes

  • Filter Manager - Filter Manager gère les filtres et la chaîne de filtres.

  • Client - Le client est l'objet qui envoie la demande à l'objet cible.

la mise en oeuvre

Nous allons créer un FilterChain , un FilterManager , une cible , un client comme divers objets représentant nos entités. AuthenticationFilter et DebugFilter représentent des filtres concrets.

InterceptingFilterDemo , notre classe de démonstration utilisera Client pour démontrer le modèle de conception de filtre d'interception.

Étape 1

Créer une interface de filtre.

Filter.java

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

Étape 2

Créez des filtres en béton.

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);
   }
}

Étape 3

Créer une cible

Target.java

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

Étape 4

Créer une chaîne de filtres

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;
   }
}

Étape 5

Créer un gestionnaire de filtres

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);
   }
}

Étape 6

Créer un client

Client.java

public class Client {
   FilterManager filterManager;

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

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

Étape 7

Utilisez le client pour démontrer le modèle de conception de filtre d'interception.

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");
   }
}

Étape 8

Vérifiez la sortie.

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

Le modèle de conception de localisateur de services est utilisé lorsque nous voulons localiser divers services à l'aide de la recherche JNDI. Compte tenu du coût élevé de la recherche de JNDI pour un service, le modèle de localisateur de service utilise la technique de mise en cache. Pour la première fois qu'un service est requis, Service Locator recherche dans JNDI et met en cache l'objet de service. Une recherche supplémentaire ou le même service via Service Locator est effectué dans son cache, ce qui améliore considérablement les performances de l'application. Voici les entités de ce type de modèle de conception.

  • Service- Service réel qui traitera la demande. La référence d'un tel service doit être examinée dans le serveur JNDI.

  • Context / Initial Context -JNDI Context, porte la référence au service utilisé à des fins de recherche.

  • Service Locator - Service Locator est un point de contact unique pour obtenir des services par recherche JNDI, mettant en cache les services.

  • Cache - Cache pour stocker les références des services pour les réutiliser

  • Client - Le client est l'objet qui appelle les services via ServiceLocator.

la mise en oeuvre

Nous allons créer un ServiceLocator , InitialContext , Cache , Service comme divers objets représentant nos entités. Service1 et Service2 représentent des services concrets.

ServiceLocatorPatternDemo , notre classe de démonstration agit en tant que client ici et utilisera ServiceLocator pour démontrer le modèle de conception de Service Locator.

Étape 1

Créer une interface de service.

Service.java

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

Étape 2

Créez des services concrets.

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";
   }
}

Étape 3

Créer InitialContext pour la recherche JNDI

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;		
   }
}

Étape 4

Créer un cache

Cache.java

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

public class Cache {

   private List<Service> services;

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

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

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

Étape 5

Créer un localisateur de services

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;
   }
}

Étape 6

Utilisez ServiceLocator pour illustrer le modèle de conception de Service Locator.

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();		
   }
}

Step 7

Verify the output.

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

The Transfer Object pattern is used when we want to pass data with multiple attributes in one shot from client to server. Transfer object is also known as Value Object. Transfer Object is a simple POJO class having getter/setter methods and is serializable so that it can be transferred over the network. It do not have any behavior. Server Side business class normally fetches data from the database and fills the POJO and send it to the client or pass it by value. For client, transfer object is read-only. Client can create its own transfer object and pass it to server to update values in database in one shot. Following are the entities of this type of design pattern.

  • Business Object - Business Service which fills the Transfer Object with data.

  • Transfer Object -Simple POJO, having methods to set/get attributes only.

  • Client - Client either requests or sends the Transfer Object to Business Object.

Implementation

We're going to create a StudentBO as Business Object,Student as Transfer Object representing our entities.

TransferObjectPatternDemo, our demo class is acting as a client here and will use StudentBO and Student to demonstrate Transfer Object Design Pattern.

Step 1

Create Transfer Object.

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;
   }
}

Step 2

Create Business Object.

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");
   }
}

Step 3

Use the StudentBO to demonstrate Transfer Object Design Pattern.

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()+" ]");
   }
}

Step 4

Verify the output.

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