Modules

Learning Goals

  • understand how mixins are used in programming
  • understand the role the Modules play in Ruby
  • use a module to create a mixin to make our code DRYer (Don’t Repeat Yourself)

Vocabulary

  • Mixin
  • Module
  • Instantiate
  • State
  • Behavior

Warm Up

Jot down your thoughts for the following questions. Be ready to share.

  • What do you know about mixins already? If little, what would you guess ‘mixins’ are all about based on the name?
  • Football players and soccer players both have unique attributes. What behaviors might they share?
  • What behaviors/abilities might be shared between an instructor and a student?

Introduction

We’re going to learn about Modules, a simple tool that will do a few completely different things in Ruby. Today we are going to talk about using them as Mixins.

Mixins

  • Mixins allow us to share behavior between objects
  • Ruby implements mixins with Modules

In Object-Oriented Programming, we can use a ‘mixin’ (or ‘mix-in’) to share methods across different classes. The implementation details of how methods are shared will depend upon the language, but often, mixins are described as being “included.” This is the way that Modules work in Ruby.

Modules

  • Modules may look like classes, but they only hold methods
  • Modules only store behavior
  • Modules do not store state
  • Modules cannot be instantiated. This means that you cannot type somethinglike MyModule.new

Let’s look at two separate classes to start exploring the idea of modules. Navigate to the examples from the lessons/modules/socials directory in the se-mod1-exercises repo.

First, we’ll experiment with the StatusUpdate class.

pry

require "./status_update.rb"
=> true

status = StatusUpdate.new("I'm learning about modules in Ruby #nbd")

status.display

status.add_comment("Oh cool!")
status.add_comment("Just wait until you learn about inheritance.")
status.add_comment("What is a module?")
status.add_comment("I'm so proud of you! Love, mom.")

status.display

status.remove_comment(3)

status.display

Now, let’s experiment with the Photo class.

require "./photo.rb"
=> true

photo = Photo.new("https://images.pexels.com/photos/2280545/pexels-photo-2280545.jpeg", "This is what I ate for breakfast #yum #hashtag")

photo.display

photo.add_comment("That looks delicious!")
photo.add_comment("Ooooh, will you cook for me?")
photo.add_comment("Brunch next Sunday?")

photo.display

photo.remove_comment(1)

photo.display

Reflection

  • What is similar/different between the two classes (StatusUpdate and Photo)?
  • What design principle(s) are we breaking with these two classes? Explain.

Modules

We can extract this duplication into a module which we’ll include within each class. Oftentimes, you’ll see modules named with the convention “-able”, like Comparable or Enumerable.

touch commentable.rb

module Commentable
  # what extracted code goes here? 
end

To get access to the methods defined in the module, you will include the module at the beginning of the class. Using include allows you to call the module methods on an instance of the class where it is included.

In status_update.rb

require "./commentable"

class StatusUpdate
  include Commentable

  # ... other code
end

In photo.rb

require "./commentable"

class Photo
  include Commentable

  # ... other code
end

Get back into Pry and try out the interaction pattern below:

require "./status_update.rb"
require "./photo.rb"

status = StatusUpdate.new("I'm learning about modules in Ruby #nbd")

status.display

status.add_comment("Oh cool!")
status.add_comment("Whoa...")

status.display

status.remove_comment(1)

status.display


photo = Photo.new("https://images.pexels.com/photos/2280545/pexels-photo-2280545.jpeg", "This is what I ate for breakfast #yum #hashtag")

photo.display

photo.add_comment("That looks delicious!")
photo.add_comment("Pancakes!")

photo.display

photo.remove_comment(2)

photo.display

Reflection

  • What just happened there?
  • What would be the benefit of a module?
  • Where else might you be able to reuse Commentable?

Key Points

  • Once a module is included in a class, any object created from that class can call the method in the module (we just treat it as if the add_comment and remove_comment methods were part of our other classes, where you call the methods on an instance of an class.)
  • Many classes can include the same module
  • Each class can include many modules

Exercise: Modules

Now it’s your turn.

Consider the following code:

class Camry
  def start
    puts "Engine on!"
  end

  def stop
    puts "Engine off!"
  end

  def drive
    puts "Back wheels go!"
  end
end
class Jeep
  def start
    puts "Engine on!"
  end

  def stop
    puts "Engine off!"
  end

  def drive
    puts "All wheels go!"
  end
end
  • Create an Engine module to extract the start and stop methods. (Yes, we know this isn’t following the “-able” naming conventions, but this is not a rule).
  • Create an Airconditionable module that is mixed into both classes. Instances of either class should be able to turn the AC on (Chilly air coming your way!) or off (Temp is fine in here.). It’s up to you what you want to name these methods.
  • Now that stop and start are extracted to the module, add back in a start or stop method in one of your class. Have it return the string “WAIIIIIIIT”. What happens when you run this method? Why?

Check for Understanding

  • In the exercise above, why didn’t we just tell you to have a ‘Vehicle’ module that could hold engine and AC? What might the benefit of those being separated be?
  • What is a module? How is it different than a class?
  • How do you allow a class to have access to module methods?

Additional Reading

Lesson Search Results

Showing top 10 results