ทับทิม - เชิงวัตถุ
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 จะพร้อมใช้งานในระหว่างการดำเนินการตามนิยามเมธอด