ทับทิม - เชิงวัตถุ

Ruby เป็นภาษาเชิงวัตถุที่บริสุทธิ์และทุกอย่างดูเหมือนกับ Ruby เป็นวัตถุ ทุกค่าใน Ruby เป็นวัตถุแม้กระทั่งสิ่งดั้งเดิมที่สุด: สตริงตัวเลขและแม้กระทั่งจริงและเท็จ แม้แต่คลาสเองก็เป็นอ็อบเจกต์ที่เป็นอินสแตนซ์ของคลาสคลาส บทนี้จะนำคุณไปสู่ฟังก์ชันหลักทั้งหมดที่เกี่ยวข้องกับ Object Oriented Ruby

คลาสถูกใช้เพื่อระบุรูปแบบของอ็อบเจ็กต์และรวมการแสดงข้อมูลและวิธีการจัดการข้อมูลนั้นเป็นแพ็กเกจเดียว ข้อมูลและวิธีการภายในคลาสเรียกว่าสมาชิกของคลาส

นิยามคลาส Ruby

เมื่อคุณกำหนดคลาสคุณจะต้องกำหนดพิมพ์เขียวสำหรับชนิดข้อมูล สิ่งนี้ไม่ได้กำหนดข้อมูลใด ๆ แต่เป็นการกำหนดความหมายของชื่อคลาสนั่นคืออ็อบเจ็กต์ของคลาสจะประกอบด้วยอะไรและการดำเนินการใดที่สามารถทำได้กับอ็อบเจ็กต์ดังกล่าว

นิยามคลาสเริ่มต้นด้วยคีย์เวิร์ด class ตามด้วย class name และคั่นด้วยไฟล์ end. ตัวอย่างเช่นเรากำหนดคลาส Box โดยใช้คีย์เวิร์ดคลาสดังนี้ -

class Box
   code
end

ชื่อต้องขึ้นต้นด้วยอักษรตัวใหญ่และตามชื่อการประชุมที่มีมากกว่าหนึ่งคำจะถูกเรียกใช้พร้อมกันโดยแต่ละคำเป็นตัวพิมพ์ใหญ่และไม่มีอักขระคั่น (CamelCase)

กำหนด Ruby Objects

คลาสจัดเตรียมพิมพ์เขียวสำหรับอ็อบเจ็กต์ดังนั้นโดยพื้นฐานแล้วอ็อบเจ็กต์จะถูกสร้างขึ้นจากคลาส เราประกาศออบเจ็กต์ของคลาสโดยใช้newคำสำคัญ. คำสั่งต่อไปนี้ประกาศสองวัตถุของคลาส Box -

box1 = Box.new
box2 = Box.new

วิธีการเริ่มต้น

initialize method เป็นวิธีการคลาส Ruby มาตรฐานและทำงานเกือบจะเหมือนกับ constructorทำงานในภาษาโปรแกรมเชิงวัตถุอื่น ๆ วิธีการเริ่มต้นมีประโยชน์เมื่อคุณต้องการเตรียมใช้งานตัวแปรคลาสบางตัวในเวลาที่สร้างอ็อบเจ็กต์ วิธีนี้อาจใช้รายการพารามิเตอร์และเช่นเดียวกับวิธีการทับทิมอื่น ๆ ที่จะนำหน้าด้วยdef คีย์เวิร์ดตามรูปด้านล่าง -

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

ตัวแปรอินสแตนซ์

instance variablesเป็นแอตทริบิวต์คลาสชนิดหนึ่งและกลายเป็นคุณสมบัติของอ็อบเจ็กต์เมื่อสร้างอ็อบเจ็กต์โดยใช้คลาส ทุกแอตทริบิวต์ของออบเจ็กต์ถูกกำหนดแยกกันและไม่มีค่าใช้จ่ายร่วมกับอ็อบเจ็กต์อื่น เข้าถึงได้โดยใช้โอเปอเรเตอร์ @ ภายในคลาส แต่เข้าถึงได้จากภายนอกคลาสที่เราใช้public วิธีการซึ่งเรียกว่า accessor methods. ถ้าเราใช้คลาสที่กำหนดไว้ข้างต้นBox จากนั้น @width และ @height เป็นตัวแปรอินสแตนซ์สำหรับ class Box

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

วิธีการเข้าถึงและตัวตั้งค่า

ในการทำให้ตัวแปรพร้อมใช้งานจากภายนอกคลาสจะต้องกำหนดตัวแปรภายใน accessor methodsวิธีการเข้าถึงเหล่านี้เรียกอีกอย่างว่าเมธอด getter ตัวอย่างต่อไปนี้แสดงการใช้วิธีการเข้าถึง -

#!/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}"

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

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

คล้ายกับวิธีการเข้าถึงซึ่งใช้ในการเข้าถึงค่าของตัวแปร Ruby ให้วิธีการตั้งค่าของตัวแปรเหล่านั้นจากภายนอกคลาสโดยใช้ setter methodsซึ่งกำหนดไว้ดังต่อไปนี้ -

#!/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}"

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

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

วิธีการอินสแตนซ์

instance methods ยังถูกกำหนดในลักษณะเดียวกับที่เรากำหนดวิธีการอื่น ๆ โดยใช้ defคีย์เวิร์ดและสามารถใช้ได้โดยใช้อินสแตนซ์คลาสเท่านั้นดังที่แสดงด้านล่าง ฟังก์ชันของพวกเขาไม่ได้ จำกัด เฉพาะการเข้าถึงตัวแปรอินสแตนซ์ แต่ยังสามารถทำสิ่งต่างๆได้มากขึ้นตามความต้องการของคุณ

#!/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}"

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

Area of the box is : 200

วิธีการและตัวแปรของคลาส

class variablesเป็นตัวแปรที่ใช้ร่วมกันระหว่างอินสแตนซ์ทั้งหมดของคลาส กล่าวอีกนัยหนึ่งคือมีหนึ่งอินสแตนซ์ของตัวแปรและเข้าถึงได้โดยอินสแตนซ์ออบเจ็กต์ ตัวแปรคลาสขึ้นต้นด้วยอักขระ @ สองตัว (@@) ตัวแปรคลาสจะต้องเริ่มต้นภายในนิยามคลาสดังที่แสดงด้านล่าง

วิธีการคลาสถูกกำหนดโดยใช้ def self.methodname()ซึ่งลงท้ายด้วยตัวคั่นท้ายและจะเรียกโดยใช้ชื่อคลาสเป็น classname.methodname ดังแสดงในตัวอย่างต่อไปนี้ -

#!/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()

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

Box count is : 2

วิธีการ to_s

คลาสใด ๆ ที่คุณกำหนดควรมี to_sวิธีการอินสแตนซ์เพื่อส่งคืนการแสดงสตริงของวัตถุ ต่อไปนี้เป็นตัวอย่างง่ายๆในการแสดงวัตถุกล่องในแง่ของความกว้างและความสูง -

#!/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}"

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

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

การควบคุมการเข้าถึง

Ruby ให้การป้องกันสามระดับในระดับวิธีการอินสแตนซ์ซึ่งอาจเป็น public, private, or protected. Ruby ไม่ใช้การควบคุมการเข้าถึงใด ๆ กับตัวแปรอินสแตนซ์และคลาส

  • Public Methods- วิธีสาธารณะใคร ๆ ก็เรียกได้ วิธีการเป็นแบบสาธารณะโดยค่าเริ่มต้นยกเว้นการเริ่มต้นซึ่งเป็นแบบส่วนตัวเสมอ

  • Private Methods- ไม่สามารถเข้าถึงวิธีการส่วนตัวหรือแม้แต่ดูจากภายนอกชั้นเรียน เฉพาะเมธอดของคลาสเท่านั้นที่สามารถเข้าถึงสมาชิกส่วนตัวได้

  • Protected Methods- เมธอดที่ได้รับการป้องกันสามารถเรียกใช้ได้โดยอ็อบเจ็กต์ของคลาสที่กำหนดและคลาสย่อยเท่านั้น การเข้าถึงจะถูกเก็บไว้ภายในครอบครัว

ต่อไปนี้เป็นตัวอย่างง่ายๆในการแสดงไวยากรณ์ของตัวปรับการเข้าถึงทั้งสามตัว -

#!/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()

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังต่อไปนี้ ที่นี่วิธีแรกเรียกว่าสำเร็จ แต่วิธีที่สองทำให้เกิดปัญหา

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

การสืบทอดคลาส

แนวคิดที่สำคัญที่สุดประการหนึ่งในการเขียนโปรแกรมเชิงวัตถุคือการถ่ายทอดทางพันธุกรรม การสืบทอดช่วยให้เรากำหนดคลาสในรูปแบบของคลาสอื่นซึ่งทำให้ง่ายต่อการสร้างและดูแลแอปพลิเคชัน

การสืบทอดยังให้โอกาสในการใช้ฟังก์ชันโค้ดซ้ำและเวลาในการใช้งานที่รวดเร็ว แต่น่าเสียดายที่ Ruby ไม่สนับสนุนการสืบทอดหลายระดับ แต่ Ruby รองรับ mixins. มิกซ์อินเปรียบเสมือนการใช้งานเฉพาะทางของการสืบทอดหลาย ๆ แบบซึ่งจะสืบทอดเฉพาะส่วนของอินเทอร์เฟซเท่านั้น

เมื่อสร้างคลาสแทนที่จะเขียนสมาชิกข้อมูลและฟังก์ชันสมาชิกใหม่ทั้งหมดโปรแกรมเมอร์สามารถกำหนดได้ว่าคลาสใหม่ควรสืบทอดสมาชิกของคลาสที่มีอยู่ คลาสที่มีอยู่นี้เรียกว่าbase class or superclassและคลาสใหม่เรียกว่า derived class or sub-class.

Ruby ยังสนับสนุนแนวคิดของคลาสย่อยนั่นคือการสืบทอดและตัวอย่างต่อไปนี้อธิบายแนวคิด ไวยากรณ์สำหรับการขยายคลาสนั้นเรียบง่าย เพียงเพิ่มอักขระ <และชื่อของซูเปอร์คลาสในคำสั่งคลาสของคุณ ตัวอย่างเช่นต่อไปนี้กำหนดคลาสBigBoxเป็นคลาสย่อยของ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()

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

Big box area is : 200

วิธีการลบล้าง

แม้ว่าคุณจะสามารถเพิ่มฟังก์ชันการทำงานใหม่ในคลาสที่ได้รับมา แต่บางครั้งคุณก็ต้องการเปลี่ยนลักษณะการทำงานของวิธีการที่กำหนดไว้แล้วในคลาสหลัก คุณสามารถทำได้ง่ายๆโดยคงชื่อเมธอดให้เหมือนเดิมและลบล้างฟังก์ชันการทำงานของเมธอดดังที่แสดงด้านล่างในตัวอย่าง -

#!/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()

ตัวดำเนินการมากเกินไป

เราต้องการให้ตัวดำเนินการ + ทำการเพิ่มเวกเตอร์ของวัตถุกล่องสองชิ้นโดยใช้ + ตัวดำเนินการ * เพื่อคูณความกว้างและความสูงของกล่องด้วยสเกลาร์และตัวดำเนินการยูนารี - เพื่อลบล้างความกว้างและความสูงของกล่อง นี่คือเวอร์ชันของคลาส Box ที่มีการกำหนดตัวดำเนินการทางคณิตศาสตร์ -

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

การแช่แข็งวัตถุ

บางครั้งเราต้องการป้องกันไม่ให้วัตถุเปลี่ยนแปลง วิธีการตรึงใน Object ช่วยให้เราสามารถทำได้โดยเปลี่ยนวัตถุให้เป็นค่าคงที่ได้อย่างมีประสิทธิภาพ วัตถุใด ๆ สามารถตรึงได้โดยการเรียกใช้Object.freeze. ไม่สามารถแก้ไขวัตถุที่แช่แข็งได้: คุณไม่สามารถเปลี่ยนตัวแปรอินสแตนซ์ได้

คุณสามารถตรวจสอบได้ว่าวัตถุนั้นถูกแช่แข็งแล้วหรือไม่ได้ใช้งาน Object.frozen?วิธีการซึ่งส่งคืนจริงในกรณีที่วัตถุถูกแช่แข็งมิฉะนั้นจะส่งคืนค่าเท็จ ตัวอย่างต่อไปนี้จะล้างแนวคิด -

#!/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}"

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

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

ค่าคงที่ของคลาส

คุณสามารถกำหนดค่าคงที่ภายในคลาสได้โดยการกำหนดค่าตัวเลขหรือสตริงโดยตรงให้กับตัวแปรซึ่งกำหนดโดยไม่ใช้ @ หรือ @@ ตามแบบแผนเราใช้ชื่อคงที่ในตัวพิมพ์ใหญ่

เมื่อกำหนดค่าคงที่แล้วคุณจะไม่สามารถเปลี่ยนค่าได้ แต่คุณสามารถเข้าถึงค่าคงที่ภายในคลาสได้โดยตรงเช่นเดียวกับตัวแปร แต่ถ้าคุณต้องการเข้าถึงค่าคงที่ภายนอกคลาสคุณจะต้องใช้ classname::constant ดังแสดงในตัวอย่างด้านล่าง

#!/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}"

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

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

ค่าคงที่ของคลาสจะได้รับการสืบทอดและสามารถแทนที่ได้เช่นเดียวกับวิธีการของอินสแตนซ์

สร้างวัตถุโดยใช้การจัดสรร

อาจมีสถานการณ์เมื่อคุณต้องการสร้างวัตถุโดยไม่เรียกตัวสร้าง initializeกล่าวคือใช้วิธีการใหม่ในกรณีเช่นนี้คุณสามารถเรียกใช้การจัดสรรซึ่งจะสร้างวัตถุที่ไม่ได้เริ่มต้นให้กับคุณดังตัวอย่างต่อไปนี้ -

#!/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}"

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

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

ข้อมูลชั้นเรียน

หากนิยามคลาสเป็นโค้ดที่เรียกใช้งานได้หมายความว่าพวกเขาดำเนินการในบริบทของอ็อบเจ็กต์บางอย่าง: self ต้องอ้างอิงบางสิ่ง มาดูกันว่ามันคืออะไร

#!/usr/bin/ruby -w

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

เมื่อดำเนินการโค้ดด้านบนจะให้ผลลัพธ์ดังนี้ -

Type of self = Class
Name of self = Box

ซึ่งหมายความว่าคำจำกัดความของคลาสถูกเรียกใช้โดยคลาสนั้นเป็นอ็อบเจ็กต์ปัจจุบัน ซึ่งหมายความว่าเมธอดใน metaclass และ superclasses จะพร้อมใช้งานในระหว่างการดำเนินการตามนิยามเมธอด