Ruby - Berorientasi Objek

Ruby adalah bahasa berorientasi objek murni dan semuanya tampak di Ruby sebagai objek. Setiap nilai di Ruby adalah sebuah objek, bahkan yang paling primitif: string, angka, dan bahkan benar dan salah. Bahkan kelas itu sendiri adalah objek yang merupakan turunan dari kelas Kelas . Bab ini akan memandu Anda mempelajari semua fungsi utama yang terkait dengan Ruby Berorientasi Objek.

Kelas digunakan untuk menentukan bentuk objek dan menggabungkan representasi data dan metode untuk memanipulasi data ke dalam satu paket yang rapi. Data dan metode di dalam kelas disebut anggota kelas.

Definisi Kelas Ruby

Saat Anda menentukan kelas, Anda menentukan cetak biru untuk tipe data. Ini tidak benar-benar mendefinisikan data apa pun, tetapi itu menentukan apa arti nama kelas, yaitu, apa objek kelas akan terdiri dan operasi apa yang dapat dilakukan pada objek semacam itu.

Definisi kelas dimulai dengan kata kunci class diikuti oleh class name dan dibatasi dengan end. Misalnya, kami mendefinisikan kelas Box menggunakan kelas kata kunci sebagai berikut -

class Box
   code
end

Nama harus dimulai dengan huruf kapital dan dengan nama konvensi yang berisi lebih dari satu kata dijalankan bersama dengan setiap kata yang menggunakan huruf besar dan tidak ada karakter pemisah (CamelCase).

Tentukan Objek Ruby

Sebuah kelas menyediakan cetak biru untuk objek, jadi pada dasarnya sebuah objek dibuat dari sebuah kelas. Kami mendeklarasikan objek kelas menggunakannewkata kunci. Pernyataan berikut mendeklarasikan dua objek kelas Box -

box1 = Box.new
box2 = Box.new

Metode menginisialisasi

Itu initialize method adalah metode kelas Ruby standar dan bekerja dengan cara yang hampir sama seperti constructorbekerja dalam bahasa pemrograman berorientasi objek lainnya. Metode inisialisasi berguna ketika Anda ingin menginisialisasi beberapa variabel kelas pada saat pembuatan objek. Metode ini dapat mengambil daftar parameter dan seperti metode ruby ​​lainnya, metode ini akan didahului olehdef kata kunci seperti yang ditunjukkan di bawah ini -

class Box
   def initialize(w,h)
      @width, @height = w, h
   end
end

Variabel Instance

Itu instance variablesadalah jenis atribut kelas dan menjadi properti objek setelah objek dibuat menggunakan kelas. Setiap atribut objek ditetapkan secara individual dan tidak memiliki nilai yang sama dengan objek lain. Mereka diakses menggunakan operator @ di dalam kelas tetapi untuk mengaksesnya di luar kelas yang kita gunakanpublic metode, yang disebut accessor methods. Jika kita mengambil kelas yang ditentukan di atasBox lalu @width dan @height adalah variabel instan untuk kelas Box.

class Box
   def initialize(w,h)
      # assign instance variables
      @width, @height = w, h
   end
end

Metode pengakses & penyetel

Untuk membuat variabel tersedia dari luar kelas, mereka harus didefinisikan di dalam accessor methods, metode pengakses ini juga dikenal sebagai metode pengambil. Contoh berikut menunjukkan penggunaan metode pengakses -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # accessor methods
   def printWidth
      @width
   end

   def printHeight
      @height
   end
end

# create an object
box = Box.new(10, 20)

# use accessor methods
x = box.printWidth()
y = box.printHeight()

puts "Width of the box is : #{x}"
puts "Height of the box is : #{y}"

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

Width of the box is : 10
Height of the box is : 20

Mirip dengan metode pengakses, yang digunakan untuk mengakses nilai variabel, Ruby menyediakan cara untuk mengatur nilai variabel tersebut dari luar kelas menggunakan setter methods, yang didefinisikan sebagai berikut -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # accessor methods
   def getWidth
      @width
   end
   def getHeight
      @height
   end

   # setter methods
   def setWidth=(value)
      @width = value
   end
   def setHeight=(value)
      @height = value
   end
end

# create an object
box = Box.new(10, 20)

# use setter methods
box.setWidth = 30
box.setHeight = 50

# use accessor methods
x = box.getWidth()
y = box.getHeight()

puts "Width of the box is : #{x}"
puts "Height of the box is : #{y}"

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

Width of the box is : 30
Height of the box is : 50

Metode Instans

Itu instance methods juga didefinisikan dengan cara yang sama seperti kita mendefinisikan metode lain yang digunakan defkata kunci dan mereka dapat digunakan menggunakan contoh kelas hanya seperti yang ditunjukkan di bawah ini. Fungsinya tidak terbatas untuk mengakses variabel instance, tetapi juga mereka dapat melakukan lebih banyak lagi sesuai kebutuhan Anda.

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # instance method
   def getArea
      @width * @height
   end
end

# create an object
box = Box.new(10, 20)

# call instance methods
a = box.getArea()
puts "Area of the box is : #{a}"

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

Area of the box is : 200

Metode dan Variabel kelas

Itu class variablesadalah variabel, yang digunakan bersama antara semua instance kelas. Dengan kata lain, ada satu contoh dari variabel dan itu diakses oleh contoh objek. Variabel kelas diawali dengan dua karakter @ (@@). Variabel kelas harus diinisialisasi dalam definisi kelas seperti yang ditunjukkan di bawah ini.

Metode kelas ditentukan menggunakan def self.methodname(), yang diakhiri dengan pembatas akhir dan akan dipanggil menggunakan nama kelas as classname.methodname seperti yang ditunjukkan pada contoh berikut -

#!/usr/bin/ruby -w

class Box
   # Initialize our class variables
   @@count = 0
   def initialize(w,h)
      # assign instance avriables
      @width, @height = w, h

      @@count += 1
   end

   def self.printCount()
      puts "Box count is : #@@count"
   end
end

# create two object
box1 = Box.new(10, 20)
box2 = Box.new(30, 100)

# call class method to print box count
Box.printCount()

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

Box count is : 2

Metode to_s

Setiap kelas yang Anda tentukan harus memiliki to_scontoh metode untuk mengembalikan representasi string dari objek. Berikut adalah contoh sederhana untuk merepresentasikan objek Kotak dalam hal lebar dan tinggi -

#!/usr/bin/ruby -w

class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # define to_s method
   def to_s
      "(w:#@width,h:#@height)"  # string formatting of the object.
   end
end

# create an object
box = Box.new(10, 20)

# to_s method will be called in reference of string automatically.
puts "String representation of box is : #{box}"

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

String representation of box is : (w:10,h:20)

Kontrol akses

Ruby memberi Anda tiga tingkat perlindungan pada tingkat metode instance, yaitu public, private, or protected. Ruby tidak menerapkan kontrol akses apa pun atas instance dan variabel kelas.

  • Public Methods- Metode publik bisa dipanggil oleh siapa saja. Metode bersifat publik secara default kecuali untuk menginisialisasi, yang selalu bersifat pribadi.

  • Private Methods- Metode privat tidak dapat diakses, atau bahkan dilihat dari luar kelas. Hanya metode kelas yang dapat mengakses anggota pribadi.

  • Protected Methods- Metode yang dilindungi hanya dapat dipanggil oleh objek dari kelas yang menentukan dan subkelasnya. Akses disimpan dalam keluarga.

Berikut adalah contoh sederhana untuk menunjukkan sintaks dari ketiga pengubah akses -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # instance method by default it is public
   def getArea
      getWidth() * getHeight
   end

   # define private accessor methods
   def getWidth
      @width
   end
   def getHeight
      @height
   end
   # make them private
   private :getWidth, :getHeight

   # instance method to print area
   def printArea
      @area = getWidth() * getHeight
      puts "Big box area is : #@area"
   end
   # make it protected
   protected :printArea
end

# create an object
box = Box.new(10, 20)

# call instance methods
a = box.getArea()
puts "Area of the box is : #{a}"

# try to call protected or methods
box.printArea()

Ketika kode di atas dijalankan, itu menghasilkan hasil sebagai berikut. Di sini, metode pertama berhasil dipanggil tetapi metode kedua memberikan masalah.

Area of the box is : 200
test.rb:42: protected method `printArea' called for #
<Box:0xb7f11280 @height = 20, @width = 10> (NoMethodError)

Warisan Kelas

Salah satu konsep terpenting dalam pemrograman berorientasi objek adalah konsep inheritance. Inheritance memungkinkan kita untuk mendefinisikan kelas dalam istilah kelas lain, yang membuatnya lebih mudah untuk membuat dan memelihara aplikasi.

Inheritance juga memberikan kesempatan untuk menggunakan kembali fungsionalitas kode dan waktu implementasi yang cepat tetapi sayangnya Ruby tidak mendukung beberapa level inheritance tetapi Ruby mendukung mixins. Mixin seperti implementasi khusus dari beberapa pewarisan di mana hanya bagian antarmuka yang diwarisi.

Saat membuat kelas, alih-alih menulis anggota data dan fungsi anggota yang benar-benar baru, pemrogram dapat menetapkan bahwa kelas baru harus mewarisi anggota kelas yang sudah ada. Kelas yang ada ini disebutbase class or superclass, dan kelas baru disebut sebagai derived class or sub-class.

Ruby juga mendukung konsep subclassing, yaitu inheritance dan contoh berikut menjelaskan konsepnya. Sintaks untuk memperluas kelas itu sederhana. Cukup tambahkan karakter <dan nama superclass ke pernyataan kelas Anda. Misalnya, berikut mendefinisikan kelas BigBox sebagai subkelas dari Box -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # instance method
   def getArea
      @width * @height
   end
end

# define a subclass
class BigBox < Box

   # add a new instance method
   def printArea
      @area = @width * @height
      puts "Big box area is : #@area"
   end
end

# create an object
box = BigBox.new(10, 20)

# print the area
box.printArea()

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

Big box area is : 200

Metode Mengganti

Meskipun Anda dapat menambahkan fungsionalitas baru di kelas turunan, tetapi terkadang Anda ingin mengubah perilaku metode yang sudah ditentukan di kelas induk. Anda dapat melakukannya hanya dengan menjaga nama metode tetap sama dan mengganti fungsionalitas metode seperti yang ditunjukkan di bawah ini pada contoh -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # instance method
   def getArea
      @width * @height
   end
end

# define a subclass
class BigBox < Box

   # change existing getArea method as follows
   def getArea
      @area = @width * @height
      puts "Big box area is : #@area"
   end
end

# create an object
box = BigBox.new(10, 20)

# print the area using overriden method.
box.getArea()

Operator Berlebihan

Kami ingin operator + melakukan penjumlahan vektor dari dua objek Box menggunakan +, operator * untuk mengalikan lebar dan tinggi Box dengan skalar, dan operator unary - untuk meniadakan lebar dan tinggi Box. Berikut adalah versi kelas Box dengan operator matematika ditentukan -

class Box
   def initialize(w,h)     # Initialize the width and height
      @width,@height = w, h
   end

   def +(other)       # Define + to do vector addition
      Box.new(@width + other.width, @height + other.height)
   end

   def -@           # Define unary minus to negate width and height
      Box.new(-@width, -@height)
   end

   def *(scalar)           # To perform scalar multiplication
      Box.new(@width*scalar, @height*scalar)
   end
end

Objek Pembeku

Terkadang, kami ingin mencegah suatu objek diubah. Metode pembekuan di Object memungkinkan kita melakukan ini, secara efektif mengubah objek menjadi sebuah konstanta. Objek apa pun dapat dibekukan dengan memohonObject.freeze. Objek yang dibekukan tidak boleh diubah: Anda tidak dapat mengubah variabel instansinya.

Anda dapat memeriksa apakah objek tertentu sudah dibekukan atau tidak digunakan Object.frozen?metode, yang mengembalikan nilai benar jika objek dibekukan, jika tidak, nilai salah dikembalikan. Contoh berikut membersihkan konsep -

#!/usr/bin/ruby -w

# define a class
class Box
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # accessor methods
   def getWidth
      @width
   end
   def getHeight
      @height
   end

   # setter methods
   def setWidth=(value)
      @width = value
   end
   def setHeight=(value)
      @height = value
   end
end

# create an object
box = Box.new(10, 20)

# let us freez this object
box.freeze
if( box.frozen? )
   puts "Box object is frozen object"
else
   puts "Box object is normal object"
end

# now try using setter methods
box.setWidth = 30
box.setHeight = 50

# use accessor methods
x = box.getWidth()
y = box.getHeight()

puts "Width of the box is : #{x}"
puts "Height of the box is : #{y}"

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

Box object is frozen object
test.rb:20:in `setWidth=': can't modify frozen object (TypeError)
   from test.rb:39

Konstanta Kelas

Anda dapat menentukan konstanta di dalam kelas dengan menetapkan nilai numerik atau string langsung ke variabel, yang ditentukan tanpa menggunakan @ atau @@. Sesuai ketentuan, kami menggunakan nama konstan dalam huruf besar.

Setelah konstanta ditentukan, Anda tidak dapat mengubah nilainya tetapi Anda dapat mengakses konstanta langsung di dalam kelas seperti variabel tetapi jika Anda ingin mengakses konstanta di luar kelas maka Anda harus menggunakan classname::constant seperti yang ditunjukkan pada contoh di bawah ini.

#!/usr/bin/ruby -w

# define a class
class Box
   BOX_COMPANY = "TATA Inc"
   BOXWEIGHT = 10
   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end
   # instance method
   def getArea
      @width * @height
   end
end

# create an object
box = Box.new(10, 20)

# call instance methods
a = box.getArea()
puts "Area of the box is : #{a}"
puts Box::BOX_COMPANY
puts "Box weight is: #{Box::BOXWEIGHT}"

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

Area of the box is : 200
TATA Inc
Box weight is: 10

Konstanta kelas diwariskan dan dapat diganti seperti metode instance.

Buat Objek Menggunakan Allocate

Mungkin ada situasi ketika Anda ingin membuat objek tanpa memanggil konstruktornya initializeyaitu menggunakan metode baru, dalam hal ini Anda dapat memanggil alokasikan , yang akan membuat objek yang tidak diinisialisasi untuk Anda seperti pada contoh berikut -

#!/usr/bin/ruby -w

# define a class
class Box
   attr_accessor :width, :height

   # constructor method
   def initialize(w,h)
      @width, @height = w, h
   end

   # instance method
   def getArea
      @width * @height
   end
end

# create an object using new
box1 = Box.new(10, 20)

# create another object using allocate
box2 = Box.allocate

# call instance method using box1
a = box1.getArea()
puts "Area of the box is : #{a}"

# call instance method using box2
a = box2.getArea()
puts "Area of the box is : #{a}"

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

Area of the box is : 200
test.rb:14: warning: instance variable @width not initialized
test.rb:14: warning: instance variable @height not initialized
test.rb:14:in `getArea': undefined method `*' 
   for nil:NilClass (NoMethodError) from test.rb:29

Informasi Kelas

Jika definisi kelas adalah kode yang dapat dieksekusi, ini menyiratkan bahwa mereka mengeksekusi dalam konteks beberapa objek: self harus mereferensikan sesuatu. Mari cari tahu apa itu.

#!/usr/bin/ruby -w

class Box
   # print class information
   puts "Type of self = #{self.type}"
   puts "Name of self = #{self.name}"
end

Ketika kode di atas dijalankan, menghasilkan hasil sebagai berikut -

Type of self = Class
Name of self = Box

Ini berarti bahwa definisi kelas dijalankan dengan kelas tersebut sebagai objek saat ini. Ini berarti bahwa metode dalam metaclass dan superclass-nya akan tersedia selama pelaksanaan definisi metode.