590 likes | 594 Views
Object-Oriented Design and Programming (Java). Topics Covered Today. 2.3 Advanced Class Design 2.3.4 Design Patterns 2.3.5 Singleton Pattern 2.3.6 Strategy Pattern. Reference. <<Design Patterns: Elements of Reusable Object-Oriented Software>> ISBN 0-201-63361-2
E N D
Topics Covered Today • 2.3 Advanced Class Design • 2.3.4 Design Patterns • 2.3.5 Singleton Pattern • 2.3.6 Strategy Pattern
Reference • <<Design Patterns: Elements of Reusable Object-Oriented Software>> • ISBN 0-201-63361-2 • <<设计模式:可复用面向对象软件的基础>>机械工业出版社 • <<Head First Design Patterns >> • <<深入浅出设计模式>> ISBN: 7-5641-0165-2
History • Patterns originated as an architectural concept by Christopher Alexander. • The Timeless Way of Building • 《建筑的永恒之道》
Alexander’s Pattern.. • Patterns describe a solution so that it can be applied many times without ever being the same.
History • In 1980s, Kent Beck and Ward Cunningham began experimenting with the idea of applying patterns to programming.
History • Design patterns gained popularity in computer science after the book <<Design Patterns: Elements of Reusable Object-Oriented Software>> by Erich Gamma, Richard Helm, Ralph Johnson, and John Vlissides (Gang of Four or GoF) was published in 1994.
Design Pattern • Design patterns describe practical solutions to common design problems that occur repeatedly in software development. • A design pattern description consists of: • Pattern name • A description of the problem that the pattern addresses • A description of the solution (for instance: class structure) • A discussion of the consequences of using the pattern
Classification • Creational Patterns (创建模式) • deal with object creation mechanisms • Structural Patterns (结构模式) • describe how classes and objects can be combined to form larger structures. • Behavioral Patterns (行为模式) • identify common communication patterns between objects and realize these patterns.
Topics Covered Today • 2.3 Advanced Class Design • 2.3.4 Design Patterns • 2.3.5 Singleton Pattern • 2.3.6 Strategy Pattern
One of a Kind • In some applications, there are classes that should only be instantiated once. • For instance: • system clock of an operating system; • one accounting system in a company; • object used for logging; • catalog of a library system • …
Answer These Questions? • How could you create a single object? • New MyClass(); • What if wanted to create another MyClass object? Could it call new on MyClass again? • Yes, if it is a public class. • And if not? • Well, if it is not a public class,only classes in the same package can instantiate it. But they can still instantiate it more than once.
Answer These Questions?. • Can we do this? • Yes. I suppose it is a class that can not be instantiated because it has a private constructor. • Is there ANY object that could use this private constructor? • The code in MyClass is the only code that can call it. public class MyClass { private MyClass() { } }
Answer These Questions?.. • What does this mean? • MyClass is a class with static method. • We can call the static method like this: MyClass.getInstance(); • Why did you use MyClass instead of some object name? • Well, getInstance() is a static method, in other words, it is a CLASS method. We need to use the class name to reference a static method. public class MyClass { public static MyClass getInstance() { } }
Answer These Questions?… • What if I put things together? Now can I instantiate a MyClass? • Yes • Can you think of a second way to instantiate an object? • MyClass.getInstance(); public class MyClass { private MyClass() { } public static MyClass getInstance() { return new MyClass(); } }
Answer These Questions?…. • Can you finish the code so that only One instance of MyClass is ever created?
Solution public class ASingletonClass { private static ASingletonClass singletonInstance; // other useful instance variable here private ASingletonClass() { } public static singleton getSingletonInstance() { if(singletonInstance == null) { singletonInstance = new ASingletonClass(); } return singletonInstance; } // other useful methods here }
Solution public class ASingletonClass { private static ASingletonClass singletonInstance; // other useful instance variable here private ASingletonClass() { } public static ASingletonClass getSingletonInstance() { if(singletonInstance == null) { singletonInstance = new ASingletonClass(); } return singletonInstance; } // other useful methods here }
Solution public class ASingletonClass { private static ASingletonClass singletonInstance; // other useful instance variable here private ASingletonClass() { } public static ASingletonClass getSingletonInstance() { if(singletonInstance == null) { singletonInstance = new ASingletonClass(); } return singletonInstance; } // other useful methods here }
Solution public class ASingletonClass { private static ASingletonClass singletonInstance; // other useful instance variable here private ASingletonClass() { } public static ASingletonClass getSingletonInstance() { if(singletonInstance == null) { singletonInstance = new ASingletonClass(); } return singletonInstance; } // other useful methods here }
Solution public class ASingletonClass{ private static ASingletonClass singletonInstance; // other useful instance variable here private ASingletonClass() { } public static ASingletonClass getSingletonInstance() { if(singletonInstance == null) { singletonInstance = new ASingletonClass(); } return singletonInstance; } // other useful methods here } Lazy instantiation(延迟实例化)is important for resource intensive objects.
Solution public class ASingletonClass { private static ASingletonClass singletonInstance; // other useful instance variable here private ASingletonClass() { } public static ASingletonClass getSingletonInstance() { if(singletonInstance == null) { singletonInstance = new ASingletonClass(); } return singletonInstance; } // other useful methods here }
Singleton Pattern Defined • The singleton pattern ensures that • only one instance of a class is created and • provides a method to access that one instance. • Singleton pattern class diagram:
public class Singleton { private static Singleton singletonInstance = new Singleton(); private Singleton() { } public static Singleton getSingletonInstance() { return singletonInstance; } } Eagerly Created Singleton Instance JVM create the unique instance of the Singleton when class is loaded.
Singleton Pattern Example • Class ICarnegieInfo contains the contact information for iCarnegie. Only one instance of class ICarnegieInfo can be created.
Code Study • Unit 2.3.5 ICarnegieInfo.java • Unit 2.3.5 ICarnegieInfoDemo.java
ICarnegieInfoDemo.java import java.io.*; public class ICarnegieInfoDemo { private static PrintWriter stdOut = new PrintWriter(System.out, true); public static void main(String[] args) { ICarnegieInfo companyInfo = ICarnegieInfo.getSingletonInstance(); stdOut.println("Name: " + companyInfo.getName()); stdOut.println("Address: " + companyInfo.getAddress()); stdOut.println("Telephone: " + companyInfo.getTelephone()); } }
Consequences • The singleton pattern has the following benefits: • A singleton class can control how and when client code accesses the single instance. • A singleton class can be easily modified • if requirements change and the application needs to limit the number of instances to a number other than one.
Topics Covered Today • 2.3 Advanced Class Design • 2.3.4 Design Patterns • 2.3.5 Singleton Pattern • 2.3.6 Strategy Pattern
Started with a simple SimUDuck app • Joe works for a company that makes a highly successful duck pond simulation game, SimUDuck. The game can show a large variety of duck species swimming and making quacking sounds. The initial designers of the system used standard OO techniques and created one Duck superclass from which all other duck types inherit.
Duck quack() swim() display() // other duck-like methods MallardDuck RedHeadDuck display() { // looks like a mallard } display() { // looks like a redhead } Duck Class Diagram The display() method is abstract, since all duck subclass looks different.
Duck quack() swim() display() fly() // other duck-like methods MallardDuck RedHeadDuck display() { // looks like a mallard } display() { // looks like a redhead } Need the Ducks to FLY good or bad??
DecoyDuck 1. Code is duplicated across subclass. 2. Runtime behavior changes are difficult.
<<Interface>> <<Interface>> Duck Flyable Quackable swim() fly() quack() display() RedHeadD MallardDuck uck RubberDuck DecoyDuck display() display() display() display() fly() fly() quack() quack() quack() Using Interface
Flying behaviors Quacking behaviors Design Principle • Identify the aspects of your application that vary and separate them from what stays the same. pull out what varies Duck Class
<<interface>> FlyBehavior fly() FlyWithWings FlyNoWay fly() { // implements duck flying } fly() { // do nothing } Design Principle. • Program to an interface, not an implementation.
<<interface>> Animal makeSound() Dog Cat makeSound() { bark();} bark() { // bark sound} makeSound() { meow(); } meow() {// meow sound} Program to interface • Programming to an implementation would be: • Dog d = new Dog(); d.bark(); • Programming to an interface/supertype would be: • Animal a = new Dog(); a.makeSound(); • Assign concrete object at run time: • Animal a = getAnimal(); a.makeSound();
MuteQuack <<interface>> QuackBehavior quack() { // do nothing } quack() Quack Squeak quack() { // implements duck quacking } quack() { // rubber duck squesk } Implementing the Duck Behavior
Integrate the Duck Behavior Each Duck has a reference to something that implements QuackBehavior interface public abstract class Duck { QuackBehavior quackBehavior; FlyBehavior flyBehavior; public Duck() { } public void performQuack() { quackBehavior.quack(); } public void performFly() { flyBehavior.fly(); } public void swim() { System.out.println("All ducks float."); } } Rather than handling the quack behavior itself, the Duck object delegates that behavior to the object referenced by quackBehavior
Implementing the Duck Subclass public class MallardDuck extends Duck { public MallardDuck() { quackBehavior = new Quack(); flyBehavior = new FlyWithWings(); } public void display() { System.out.println(" I am a real mallard duck "); } }
Define a set of Fly Behaviors //FlyBehavior.java public interface FlyBehavior { void fly(); } // FlyWithWings.java public class FlyWithWings implements FlyBehavior { public void fly() { System.out.println("I am flying!"); } } // FlyNoWay.java public class FlyNoWay implements FlyBehavior { public void fly() { System.out.println("I can not fly."); } }
Define a set of Quack Behaviors // QuackBehavior.java public interface QuackBehavior { void quack(); } // Quack.java public class Quack implements QuackBehavior { public void quack() { System.out.println("Quack"); } } // MuteQuack.java public class MuteQuack implements QuackBehavior { public void quack() { System.out.println("<<silence>>"); } } // Squeak.java public class Squeak implements QuackBehavior { public void quack() { System.out.println("Squeak"); } }
Test Class // MiniDuckSimulator public class MiniDuckSimulator { public static void main (String[] args){ Duck mallard = new MallardDuck(); mallard.performQuack(); mallard.performFly(); } } Quack I am flying!
Setting Behavior Dynamically public abstract class Duck { QuackBehavior quackBehavior; FlyBehavior flyBehavior; public Duck() { } public void performQuack() { quackBehavior.quack(); } public void performFly() { flyBehavior.fly(); } public void swim() { System.out.println("All ducks float."); } public void setFlyBehavior(FlyBehavior fb) { flyBehavior = fb; } public void setQuackBehavior(QuackBehavior qb) { quackBehavior = qb; } }
ModelDuck public class ModelDuck extends Duck { public ModelDuck() { flyBehavior = new FlyNoWay(); quackBehavior = new MuteQuack(); } public void display() { System.out.println("I am a model duck"); } }
MiniDuckSimulator2 //MiniDuckSimulator2 public class MiniDuckSimulator2 { public static void main (String[] args) { Duck mallard = new MallardDuck(); mallard.performQuack(); mallard.performFly(); Duck model = new ModelDuck(); model.performFly(); model.performQuack(); model.setFlyBehavior(new FlyWithWings()); model.setQuackBehavior(new Squeak()); model.performFly(); model.performQuack(); } } Quack I am flying! I can not fly. <<silence>> I am flying! Squeak