Skip to content

Latest commit

 

History

History
119 lines (68 loc) · 6.74 KB

README.md

File metadata and controls

119 lines (68 loc) · 6.74 KB

Practical Object Oriented Design in Ruby

Personal Notes on Sandy Metz's Practical Object Oriented Design in Ruby

Chapter 1: Object-Oriented Design

Application that are easy to change are apleasure to write and a joy to extend . They're flexible and adaptable. Applications tha tresist change are just the opposite; every change is expensive and each makes the next cost more.

The object and the message

Object oriented design is made up of two main concepts: the parts and the interaction between the parts. These are called the Objects and the Messages

The thing with interaction is that it requires the sender and reciever to know things about each other. This knowledge creates dependency between the objects which is not desired. The objective is to arrange dependencies such that objects can tolerate change.

A practical definition of design

You must not only write code for the feature you plan to deliver today, you must also create code that is amenable to being changed later.

you must combine an overall understanding of your application's requirements with knowledge of the costs and benefits of design alternatives and then devise an arrangement of code that is cost effective in te present and will continue to be so in the future

The purpose of design is to allow you to do design later and its primary goal is to reduce the cost of change.

Judging design

If design takes half your time htis month and does nt start returning dividents for ayear, it may not be wirth it. When the acto of design prevents software from being delivered on time, you have lost. Delivering half of a well-designed application might be the same as delivering no application at all. Hoever, if design takes half of your time this morning, payos that itme back this afternoon, and then continues to provide benefits for the lifetime of the applicaiton, you get a kind of daily compounding interest on your time; this design effor pays forever.

Chapter 2: Designing classes with a Single Responsibility

Your goal is to model your application, using classes, such that it does what it is supposed to do right now and is also easy to change later.

Design is more the art of preserving changeability than it is the act of achieving perfeciton.

Organizing code to allow for easy changes

What does "easy changes" mean?

  • A change you make should have no unexpected side effects. It should be obvious and apparent what it does and the effect it has in your application
  • Small changes in requirements require correspondingly small changes in code
  • Easy to use. Easy to expand.

To achieve this, your code should be T.R.U.E:

  1. Transparent: I immediatelly think of callbacks in Rails applications. They are so automagical that you don't know what's happening under the hood. They're useful but should be used with caution and maybe they shouldn't handle critical behaviour. _The consequences of change should be obvious in the code that is changing and in distant code relies upon it.

  2. Reasonable: The cost of any change should be proportional to the benefits the change achieves

  3. Usable: The code you already wrote should be usable in new and unexpected contexts.

  4. Exemplary: Developers copy the practices and methodologies other developers already coded. Be mindful of this as you and other developers will reuse the same practices.

Why single responsibility matters

Applications that are easy to change consist of classes that are easy to reuse.

When classes have more than one responsibility they become difficult to change and reuse. You should strive to maintain a single responsibility for each class and don't duplicate code. This will lead to increased bugs and aditional maintenance.

The SRP or "Single responsibility Principle" should not be taken literaly. If you do, classes will do to small/unique of a thing. Instead think of it as a class being highly cohesive: everithing the class does be highly related to its purpose.

When the future cost of doing nothing is sthe same as the current cost, postpone the decision. Make the decision only when you must with the information you have at the time.

Writing code that embraces change

Depend on behavior, not data

  1. Hide instance variables

Always wrap instance variables in accessor methods instead of directly referring to variables.

  1. Hide data structures

If you can control the input, pass in a useful object, bu tif you are compelled to take a messy structure, hide the mess even from yourself.

  1. Enforce Single Responsibility Everywhere

Methods, like classes, should have a single responsibility.

Methods with a single responsability:

  1. Expose previously hidden qualities
  2. Avoid the need for comments: If a bit of code inside a method nees a comment, extract that bit into a separate method.
  3. Encourage reuse. Make code exemplary since other programmers will copy your practices.
  4. Are easy to move to another class. Well separated bits of code can easily be refactored into their own classes.

Chapter 3: Managing dependencies

Understanding dependencies

An object depends on another object if, when one object changes the othe rmight be forced to change in turn.

Writing loosely coupled code

Inject dependencies

(See example code)

In summary: Pass a wheel instance to the class gear rather than passing the arguments gear needs to be initialized. The new weel argument responds to diameter. This way, any thing that is diameterable, as in, it resopnds to diameter can be used in gear

Isolate dependencies

(See example code)

Think of every dependency as an alien bacterium that's trying to infect your class. Give your class a vigorous immune system; quarantine each dependency. Dependencies are foreign invaders that represent vulnerabilities, and the should be concise, explicit and isolated.

Isolate vulnerable external messages

(See example code)

Remove argument-order dependencies

(See example code)

if your defaults are more than simple nubmers or stirngs, implement a defaults method

Isolate multiparameter initialization

(See example code)

When you cannot change the arguments from order-dependent to order-independent, ISOLATE!

Managing dependency direction

depend on things that change less often than you do

  • Some classes are more likely to change than others
  • Concrete classes are more likely to change than abstract classes (this makes me thing of rails ::Base modules)
  • Chaning a class that has many dependents will result in widespread consequences

Depending on an abstraction is always better than depending on a concretion