CSC/ECE 517 Fall 2011/ch4 4h sv: Difference between revisions
Line 5: | Line 5: | ||
__TOC__ | __TOC__ | ||
==Overview== | ==Overview== |
Revision as of 22:29, 17 October 2011
Design Patterns in Ruby
"In software engineering, a design pattern is a general reusable solution to a commonly occurring problem within a given context in software design".
A Design Pattern is a template to solve a problem which can be used in many different situations. It names, abstracts and identifies the key aspects of a common design structure that makes it useful for creating a reusable object - oriented design. Design Patterns in Object - Oriented Languages help show the relationship between the different classes and objects in the program. Design Patterns are tools for building software.
Overview
There are different types of design patterns:
- Singleton Design Pattern, which is used to have only one instance of a class.
- Adapter Design Pattern, which enables classes with incompatible interfaces to work together.
- Command Design Pattern, which enables to pass around the code that needs to be executed later.
- Algorithm Strategy Pattern, which helps choose an algorithm to fulfill a task based on some "parameter" of the situation.
Singleton Pattern
The Singleton design pattern is used to restrict the instantiation of a class to only one instance which is globally available. This is used in situations where a user needs an instance of the class to be available in various parts of the application, being available for logging functionality, communictaion with external systems and database access etc. The Singleton pattern is available as a mixin in the Ruby library. Including it in the code makes the new method private and provides an instance method used to create or access the single instance.
Below is an illustration of the implementation of Singleton Design Pattern in Ruby:
require 'singleton' class Example attr_accessor :val include Singleton end
In the above declaration of a class, including the Singleton module makes the class's new method private. To create an object of that class, the users call the instance method, which returns a singleton instance of the class.
a = Example.instance b = Example.instance
Here, the instances a and b of the class Example are essentially the same object. When a value of 007 is assigned to a, the value of b is also the same.
a.val = 007 puts b.val => 007
Below is an illustration of the Singleton Design Pattern without using the Singleton library. Here, we can observe that the class Logger's new method has been explicitly declared to be private.
class Logger def initialize @log = File.open("log.txt", "a") end @@instance = Logger.new def self.instance return @@instance end def log(msg) @log.puts(msg) end private_class_method :new end Logger.instance.log('message 1')
In this code example, inside class Logger we create instance of the very same class Logger and we can access that instance with class method Logger.instance whenever we need to write something to the log file using the instance method "log". In the initialize method we just opened a log file for appending, and at the end of Logger class, we made method "new" private so that we cannot create new instances of class Logger. And, that is Singleton Pattern: only one instance, globally available.
Adapter Pattern
An Adapter Design Pattern, also known as the Wrapper Pattern enables classes with incompatible interfaces to work together, by providing the users with its interface. This is achieved by:
- “Wrapping” its own interface around the interface of a pre-existing class.
- Besides, it may also translate data formats from the caller to a form needed by the callee. Example: If the caller stores boolean value in terms of integers but the callee needs the values as 'true/false', the adapter pattern would extract the right value from the caller and pass it on to the callee. This ensures that the caller and callee can work together.
The purpose of an adapter is “to convert the interface of a class into another interface clients expect. Adapter lets classes work together that couldn’t otherwise because of incompatible interfaces.”
Below is an illustration of the Adapter Pattern:
Types of Adapter Patterns
Based on the structure, there are two types of Adapter Patterns:
Object Adapter Pattern
The Adapter Pattern has an instance of the classit "wraps" and makes calls to this wrapped object alone.
Class Adapter Pattern
Polymorphic Interfaces are used by this type of pattern, where the interface that is expected and the interface that is available are both inherited or implemented. This is used in situations where the class providing the services you need does not provide the interface required.
Command Pattern
At times, it is needed to pass around code that needs to be executed later. This is achieved by employing the Command Design Pattern which is used to factor out the action code of a class into its own object. The Command Pattern is implemented using Closures. A Command class holds an object, method along with the information needed to call a method at a later time. This information includes the method name with the values for the method parameters. The 'Call' method of Ruby brings the different parts together when needed.
This is illustrated using the following example:
One example is a check at a restaurant.
- The waiter/waitress takes an order from a customer and writes it on a check.
- The check is then queued for the cook, who prepares the food as requested by the customer.
- The check is later returned to the server, who uses it to bill the customer.
In this example, the check has nothing to do with the menu. In principle, the same checks could be used at any restaurant.
The command pattern is made up of a client, invoker and receiver. A client creates an object of the Command class and provides the information required to call the method at a later time. The invoker decides when the method should be called. The receiver is the class object which contains the code of the method. The usage of command objects makes it easier to construct components which need to execute methods at a later time without having knowledge about the method's owner or parameters.
This Pattern can be implemented using the Proc objects, which is a callable block of code that closes over the variables in scope when it was created. This gives us a concise implementation of Command Pattern.
Below is an illustration of the Command Design Pattern, using Proc Objects:
count = 0 commands = [] (1..10).each do |i| commands << proc { count += i } end puts "Count is initially #{count}" commands.each { |cmd| cmd.call } puts "Performed all commands. count is #{count}"
Another important use of the Command Pattern is to enable the client undo what he has already done, or redo what has been undone by the user. This is done by maintaining a history of the commands executed. As and when the user makes changes, the system creates command after command, executing each command immediately to effect the change. Every undo - able command holds two methods - the execute and the unexecute method. The commands are stored in a list and when the user decides to undo a change, the last command on the list is executed to undo the change. The changes can be undone, by going back the history of commands. The redo is done in the similar way where the the commands are re-executed beginning from the last change that what undone to reapply the changes undone. A simple example of the undo - redo use of a Command Pattern is a Calculator with many undo - redo options.
Algorithm Strategy Pattern
The Strategy Pattern helps choose an algorithm to accomplish a task based on some "parameters" of the situaton. Also known as the Policy Pattern, it enables selection of algorithms at runtime. This pattern allows the algorithm to vary irresepctive of the user that uses it.
The strategy pattern "defines a family of algorithms, encapsulates each one, and makes them interchangeable".
Consider as an example, a class that that converts among different types of file formats like jpeg, jif, png etc. We can write a case statement to choose what algorithm has to be employed for each type of format. Another example could be performing validation on incoming data. Here we can select a validation algorithm based on the type and source of data.
A Strategy Pattern is best implemented using Proc Objects. Below is an illustration of it:
class RoutePlanner attr_accessor :strategy def go(pointA, pointB) strategy.call(*args) end def fastest … end def shortest … end end
Here, we have a class which helps us plan the route between two points A and B, depending upon whether we want the shortest path or the fastest path.
ctx = RoutePlanner.new if on_foot ctx.strategy = RoutePlanner.method(:shortest) else ctx.strategy = RoutePlanner.method(:fastest) ctx.go(pointA, pointB)
Here, the path or strategy chosen is based on an if condition involving the value of the variable on_foot which is not known until runtime. Therefore, the algorithm that will be made use of to reach B starting from A, is decided at run time. The strategy that will be chosen depends on many factors which change dynamically.
Differences between Command and Strategy Pattern
- A Command Pattern encapsulates a single action. A command object has a single method with a generic signature associated with it.
- A Strategy Pattern, enables us to customize an algorithm, deciding which algorithm has to be made use of depending on a number of dynamically changing factors. A given strategy has many methods associated with it.