##Objectives
- Understand difference between objects and classes
- Understand how objects are referenced
- Understand getters and setters
- Understand
attr_writer,attr_reader,attr_accessor - Understand instance variables and instance methods
- Understand class variables and class methods
- Utilize the
selfkeyword - Understand
Private, andPublicmethods - Understand method chaining in a class
Let's create our first class.
person.rb
class Person
endThis defines a class definition of a Person. The class keyword denotes the begining of a class definition.
To create a new instance of our class we write the following:
Person.newA particular instance of a class is a called an object. In general, languages that use objects as a primary means of data abstraction are said to be Object Oriented Programming (OOP) languages.
What is an object in ruby? Basically everything that isn't a keyword.
However, this can cause you some headaches if you're not careful.
Imagine we had the following
arr1 = [1,2,3]
arr2 = arr1
arr1 << 4
#=> [1,2,3,4]
arr2
#=> [1,2,3,4]Wow, the second array completely changed. That's because arr2 was a reference to arr1. Both variables represented the same object. The way around this is to copy the object.
arr1 = [1,2,3]
arr2 = Array.new(arr1)
arr1 << 4
#=> [1,2,3,4]
arr2
#=> [1,2,3]In our class definition we can make use of an initialize method, which is run when a new instance of the class is created.
class Person
def initialize
puts "A new person was created"
end
endWe can also make use of instance variables that are defined for each particular object and are available throughout other methods in the object. These variables are prefixed by an @ symbol, i.e. @my_var.
class Person
def initialize(name)
@name = name
end
def greet
puts "Hello! My name is #{@name}."
end
endNow, when we create a new Person we are required to specify the name of the person.
person = Person.new("John")
person.greet
=> Hello! My name is John.Having created an instance variable in our object, we might want to read that outside our object. However, we have to define a method that will act as an interface for reading for this variable called a Getter Method.
class Person
def initialize(name)
@name = name
end
def name
@name
end
endNow we can read or get the name outside the object.
person = Person.new("Jane")
person.name
=> "Jane"Similarly, we may need to change or set an instance variable from outside the object, so we create a method called a setter method.
class Person
def initialize(name)
@name = name
end
def name
@name
end
def name=(other)
@name = other
end
endWe can now get and set the name of a person using the methods we created for @name.
person = Person.new("Samantha")
person.name
=> "Samantha"
person.name = "Sam"
person.name
=> "Sam"In ruby, the practice of creating a getter method is so common there is a shorthand that can be used at the top of a class definition called attr_reader.
class Person
attr_reader :name
def initailize(name)
@name = name
end
def name=(other)
@name = other
end
endSimilarly, we can do the same with the setter method using attr_writer
class Person
attr_reader :name
attr_writer :name
def initialize(name)
@name = name
end
endMoreover, we have a shorthand for telling our class to create both a getter and a setter method called attr_accessor.
class Person
attr_accessor :name
def initialize(name)
@name = name
end
endWe just created instance variables, which have different values depending on the object instance. Class variables share the same value across the entire class. Also, we don't need to create an instance in order to access class variables.
Let's first create a variable associated with our class. Using the syntax @@var_name designates a class variable.
class Person
attr_accessor :name
@@population = 0
def initialize(name)
@name = name
@@population += 1
end
endWe can access the value without creating any people.
Person.population
#=> 0If we create a few people, we see the following
Person.new("John")
Person.new("Jane")
Person.population
#=> 2What about class methods? We can also create a method that belongs to a class.
class Person
attr_accessor :name
@@population = 0
def initialize(name)
@name = name
@@population += 1
end
def self.print_population
puts "There are #{self.population} people"
end
endIn most cases, inside an instance method, self refers to the object, but when used in the context of a method name, self refers to the class itself`.
Also, note that self can be used in instance methods to refer to particular object in use, i.e. self.var_name instead of @var_name.
If we create a class Person with a name attribute and use attr_accessor to create the getters and setters as follows
class Person
attr_accessor :name
def initialize(name)
@name = name
end
endthen anyone can read and access Person#name.
person1 = Person.new("John")
person1.name
=> "John"We can change this using the private keyword. Everything under the private keyword is private to outside users. Only the instance methods can use the getter and setter.
class Person
def initialize(name)
@name = name
end
private
attr_accessor :name
endWe can also add private methods by defining new methods below the private keyword
class Person
def initialize(name)
@name = name
end
private
attr_accessor :name
def make_call
puts "Calling friends"
end
endNote that we can create a public method that calls a private method, because we are within the class.
class Person
def initialize(name)
@name = name
end
def call
make_call if name
end
private
attr_accessor :name
def make_call
puts "Calling friends"
end
endWhat if I wanted to create a class that had chainable methods calling many methods in one line.
class Person
def initialize(name)
@name = name
end
def greet
puts "Hello I am #{@name}."
puts "What is your name?"
@other = gets.chomp
puts "Nice to meet you, #{@other}."
end
def thank
puts "Thank you for coming."
end
def farewell
puts "Farewell, #{@other}"
end
endTrying to do
person1 = Person.new("john")
person1.greet.thank.farewell
#=> NoMethodError: undefined method `thank' for nil:NilClassto achieve this we have to return a reference to the object after each method
class Person
def initialize(name)
@name = name
end
def greet
puts "Hello I am #{@name}."
puts "What is your name?"
@other = gets.chomp
puts "Nice to meet you, #{@other}."
self
end
def thank
puts "Thank you for coming."
self
end
def farewell
puts "Farewell, #{@other}"
self
end
end