CSC/ECE 517 Fall 2010/ch7 7a ed: Difference between revisions
No edit summary |
No edit summary |
||
Line 29: | Line 29: | ||
=Design Pattern Implementations= | =Design Pattern Implementations= | ||
==( | ==Strategy Pattern== | ||
Using closures, dynamic languages are able to simply pass in blocks of code that implement the strategy to be used. | |||
For example, the Ruby sort method accepts a closure block, with which you can pass in the sorting metric. | |||
>> list = ["10", "2", "5"] | |||
=> ["10", "2", "5"] | |||
>> list.sort{|a, b| a.length <=> b.length} # length strategy | |||
=> ["5", "2", "10"] | |||
>> list.sort{|a, b| a.to_i <=> b.to_i} # integer strategy | |||
=> ["2", "5", "10"] | |||
>> list.sort{|a, b| a <=> b} # string strategy | |||
=> ["10", "2", "5"] | |||
These strategies can also be stored and used later or chosen based on a conditional. For example: | |||
length_strategy = lambda {|a, b| a.length <=> b.length} | |||
list.sort(&length_strategy) | |||
==(reflection)== | ==(reflection)== |
Revision as of 06:27, 2 December 2010
Novel Implementations of Design Patterns in Dynamic Languages
Dynamic languages such as Ruby and Javascript allow many novel implementations of the design patterns we cover in previous chapters, which would not be possible or feasible in complied languages such as Java and C. Language features only available in dynamic languages give developers an increased amount of flexibility in how they can implement design patterns. These language features and the some examples of the implementations they allow are detailed in this section, with a focus on implementations in the Ruby language.
Language Features
"These behaviors can be emulated in nearly any language of sufficient complexity, but dynamic languages provide direct tools to make use of them" [1]. Some of the most helpful tools are as follows:
Closures
Closures are a feature that allow blocks of code to be passed around as arguments to functions. Closures are also known as lambdas or procs in Ruby. The code in a closure has access to all variables in the scope where it was created, and it can have additional data passed in as parameters to the closure from the caller. [2]
Runtime Object Modification
Dynamic languages allow metaprogramming, or the ability of the program to modify itself, creating or adding to objects and classes. This is explored further in section 4G.
eval
In dynamic languages, any arbitrary code can be executed dynamically at runtime. This can include code that is generated from user input. This is a subset of metaprogramming. Eval is explored further in section 3E.
Reflection
Reflection is the ability of a language to inspect the different methods exist on an object. Reflection is explored further in section 1C.
Dynamically Named Methods
Dynamic languages often allow objects to handle method calls to methods that do not exist. In ruby, method_missing provides this functionality.
Design Pattern Implementations
Strategy Pattern
Using closures, dynamic languages are able to simply pass in blocks of code that implement the strategy to be used.
For example, the Ruby sort method accepts a closure block, with which you can pass in the sorting metric.
>> list = ["10", "2", "5"] => ["10", "2", "5"] >> list.sort{|a, b| a.length <=> b.length} # length strategy => ["5", "2", "10"] >> list.sort{|a, b| a.to_i <=> b.to_i} # integer strategy => ["2", "5", "10"] >> list.sort{|a, b| a <=> b} # string strategy => ["10", "2", "5"]
These strategies can also be stored and used later or chosen based on a conditional. For example:
length_strategy = lambda {|a, b| a.length <=> b.length} list.sort(&length_strategy)
(reflection)
Command Pattern
Using eval or send, one can implement the command pattern. This is implemented in Ruby on Rails' ActionController. A web application user enters a URL, which is the a form of a command. The URL typically contains the name of the controller and the action (command) that is to be executed on that controller. The application checks to see if the a method with the action name exists in the controller and runs that method if so.
Object Relational Mapping
Ruby on Rails' ActiveRecord implementation of the Object Relational Mapping (ORM) pattern uses runtime object modification and dynamically named methods to achieve a very clean implementation.
Runtime Object Modification
Every ActiveRecord model contains accessors for each field in the corresponding database table. These accessors are created at runtime and do not require the programmer to manually create and update accessors for each field in the database. It does this by inspecting the database and adding accessor methods to the model's class for each field. [3] This allows for much cleaner model code.
Dynamically Named Methods
By overriding Ruby's method_missing method in models, ActiveRecord can look for certain patterns in method calls on the models where no method is explicitly defined. One such pattern is find_all_by_FIELD_NAME
, where FIELD_NAME can be any field in the model's database table. Using these convenience methods, the models can be less cluttered, and Rails developers can write code that is easier to read.
References
- Wikipedia. Dynamic programming language. 2 October 2010. http://en.wikipedia.org/wiki/Dynamic_programming_language
- Cornford, Richard. Javascript Closures. March 2004. http://jibbering.com/faq/notes/closures
- 37Signals. Active Record — Object-relation mapping put on rails. 27 June 2008. http://ar.rubyonrails.org