Rails เพิ่มหลายสิ่งหลายอย่างให้กับวัตถุในตัวของ Ruby นี่คือสิ่งที่บางคนเรียกว่า "ภาษาถิ่น" ของ Ruby และเป็นสิ่งที่ช่วยให้นักพัฒนา Rails สามารถเขียนบรรทัดต่างๆ เช่น 1.day.ago .
วิธีการพิเศษเหล่านี้ส่วนใหญ่อยู่ใน ActiveSupport วันนี้ เราจะมาดูวิธีที่อาจไม่ค่อยมีใครรู้จักซึ่ง ActiveSupport เพิ่มลงใน Class โดยตรง:descendants . เมธอดนี้ส่งคืนคลาสย่อยทั้งหมดของคลาสที่เรียก ตัวอย่างเช่น ApplicationRecord.descendants จะส่งคืนคลาสในแอปของคุณที่สืบทอดมาจากมัน (เช่น ทุกรุ่นในแอปพลิเคชันของคุณ) ในบทความนี้ เราจะมาดูวิธีการทำงาน เหตุผลที่คุณอาจต้องการใช้ และวิธีเพิ่มวิธีการที่เกี่ยวข้องกับการสืบทอดในตัวของ Ruby
การสืบทอดในภาษาเชิงวัตถุ
อันดับแรก เราจะให้การทบทวนอย่างรวดเร็วเกี่ยวกับรูปแบบการสืบทอดของ Ruby เช่นเดียวกับภาษาเชิงวัตถุ (OO) อื่นๆ Ruby ใช้วัตถุที่อยู่ในลำดับชั้น คุณสามารถสร้างคลาส จากนั้นเป็นคลาสย่อยของคลาสนั้น จากนั้นเป็นคลาสย่อยของคลาสย่อยนั้น และอื่นๆ เมื่อเดินขึ้นลำดับนี้ เราก็ได้รายชื่อบรรพบุรุษ Ruby ยังมีคุณสมบัติที่ดีที่ ทั้งหมด เอนทิตีเป็นวัตถุเอง (รวมถึงคลาส จำนวนเต็ม และแม้แต่ศูนย์) ในขณะที่ภาษาอื่นบางภาษามักใช้ "พื้นฐาน" ที่ไม่ใช่วัตถุจริง โดยปกติแล้วเพื่อประสิทธิภาพ (เช่น จำนวนเต็ม คู่ บูลีน ฯลฯ ฉัน' มองมาที่คุณ Java)
Ruby และภาษา OO ทั้งหมดต้องติดตามบรรพบุรุษเพื่อที่จะรู้ว่าจะหาวิธีการได้ที่ไหนและภาษาใดมีความสำคัญกว่า
class BaseClass
def base
"base"
end
def overridden
"Base"
end
end
class SubClass < BaseClass
def overridden
"Subclass"
end
end
ที่นี่เรียก SubClass.new.overridden ให้เรา "SubClass" . อย่างไรก็ตาม SubClass.new.base ไม่มีอยู่ในคำจำกัดความ SubClass ของเรา ดังนั้น Ruby จะตรวจสอบบรรพบุรุษแต่ละคนเพื่อดูว่าวิธีใดนำวิธีการไปใช้ (ถ้ามี) เราสามารถดูรายชื่อบรรพบุรุษได้โดยเพียงแค่เรียก SubClass.ancestors . ใน Rails ผลลัพธ์จะเป็นดังนี้:
[SubClass,
BaseClass,
ActiveSupport::Dependencies::ZeitwerkIntegration::RequireDependency,
ActiveSupport::ToJsonWithActiveSupportEncoder,
Object,
PP::ObjectMixin,
JSON::Ext::Generator::GeneratorMethods::Object,
ActiveSupport::Tryable,
ActiveSupport::Dependencies::Loadable,
Kernel,
BasicObject]
เราจะไม่วิเคราะห์รายชื่อทั้งหมดที่นี่ เพื่อจุดประสงค์ของเรา แค่สังเกตว่า SubClass อยู่ที่ด้านบนสุด โดยมี BaseClass ด้านล่างมัน นอกจากนี้ โปรดทราบว่า BasicObject อยู่ที่ด้านล่าง นี่คือออบเจ็กต์ระดับบนสุดใน Ruby ดังนั้นจึงจะอยู่ที่ด้านล่างของสแต็กเสมอ
โมดูล (a.k.a. 'Mixins')
สิ่งต่างๆ จะซับซ้อนขึ้นเล็กน้อยเมื่อเราเพิ่มโมดูลลงในมิกซ์ โมดูลไม่ใช่บรรพบุรุษในลำดับชั้นของคลาส แต่เราสามารถ "รวม" ไว้ในคลาสของเราได้ ดังนั้น Ruby จึงต้องรู้ว่าเมื่อใดควรตรวจสอบโมดูลสำหรับเมธอด หรือแม้แต่โมดูลใดที่จะตรวจสอบก่อนในกรณีที่มีหลายโมดูลรวมอยู่ด้วย .
บางภาษาไม่อนุญาตให้มี "การสืบทอดหลายรายการ" ประเภทนี้ แต่ Ruby ยังก้าวไปอีกขั้นโดยให้เราเลือกตำแหน่งที่จะแทรกโมดูลลงในลำดับชั้น ว่าเรารวมหรือเพิ่มโมดูลไว้ข้างหน้าหรือไม่
กำลังเตรียมโมดูล
โมดูลที่เติมหน้า ตามชื่อที่แนะนำ จะถูกแทรกลงในรายการบรรพบุรุษก่อนคลาส โดยทั่วไปจะแทนที่เมธอดใดๆ ของคลาส นอกจากนี้ยังหมายความว่าคุณสามารถเรียก "super" ในเมธอดของโมดูลที่นำหน้าเพื่อเรียกเมธอดของคลาสดั้งเดิมได้
module PrependedModule
def test
"module"
end
def super_test
super
end
end
# Re-using `BaseClass` from earlier
class SubClass < BaseClass
prepend PrependedModule
def test
"Subclass"
end
def super_test
"Super calls SubClass"
end
end
บรรพบุรุษสำหรับ SubClass ตอนนี้มีลักษณะดังนี้:
[PrependedModule,
SubClass,
BaseClass,
ActiveSupport::Dependencies::ZeitwerkIntegration::RequireDependency,
...
]
ด้วยรายชื่อบรรพบุรุษใหม่นี้ PrependedModule . ของเรา ตอนนี้อยู่ในบรรทัดแรก หมายความว่า Ruby จะมองหาวิธีการใดๆ ที่เราเรียกใช้ใน SubClass ที่นั่นก่อน . นี้ด้วย หมายความว่าถ้าเราเรียก super ภายใน PrependedModule เราจะเรียกเมธอดใน SubClass :
> SubClass.new.test
=> "module"
> SubClass.new.super_test
=> "Super calls SubClass"
รวมโมดูล
โมดูลที่รวมไว้จะถูกแทรกลงในบรรพบุรุษ หลังจาก ห้องเรียน. วิธีนี้เหมาะอย่างยิ่งสำหรับวิธีการสกัดกั้นที่คลาสพื้นฐานจะจัดการ
class BaseClass
def super_test
"Super calls base class"
end
end
module IncludedModule
def test
"module"
end
def super_test
super
end
end
class SubClass < BaseClass
include IncludedModule
def test
"Subclass"
end
end
ด้วยการจัดเรียงนี้ บรรพบุรุษของ SubClass จะมีลักษณะดังนี้:
[SubClass,
IncludedModule,
BaseClass,
ActiveSupport::Dependencies::ZeitwerkIntegration::RequireDependency,
...
]
ตอนนี้ SubClass เป็นจุดแรกของการโทร ดังนั้น Ruby จะดำเนินการเฉพาะเมธอดใน IncludedModule หากไม่มีอยู่ใน SubClass . ส่วน super , การเรียกใด ๆ ไปยัง super ใน SubClass จะไปที่ IncludedModule อย่างแรกในขณะที่เรียกใช้ super ภายใน IncludedModule จะไปที่ BaseClass .
ในอีกทางหนึ่ง โมดูลที่รวมไว้จะอยู่ระหว่างคลาสย่อยและคลาสพื้นฐานในลำดับชั้นของบรรพบุรุษ นี่หมายความว่าสามารถใช้เพื่อ 'สกัดกั้น' วิธีที่อาจได้รับการจัดการโดยคลาสพื้นฐาน:
> SubClass.new.test
=> "Subclass"
> SubClass.new.super_test
=> "Super calls BaseClass"
ด้วยเหตุนี้ "สายการบังคับบัญชา" นี้ Ruby จึงต้องติดตามบรรพบุรุษของคลาส กลับไม่เป็นความจริงแม้ว่า สำหรับคลาสใดคลาสหนึ่ง Ruby ไม่จำเป็นต้องติดตามลูกๆ หรือ "ผู้สืบทอด" เพราะจะไม่ต้องการข้อมูลนี้เพื่อดำเนินการเมธอด
การจัดลำดับบรรพบุรุษ
ผู้อ่านที่ชาญฉลาดอาจรู้ว่าถ้าเราใช้หลายโมดูลในชั้นเรียน ลำดับที่เรารวม (หรือเติมหน้า) โมดูลเหล่านี้อาจให้ผลลัพธ์ที่แตกต่างกัน ตัวอย่างเช่น ขึ้นอยู่กับเมธอด คลาสนี้:
class SubClass < BaseClass
include IncludedModule
include IncludedOtherModule
end
และคลาสนี้:
class SubClass < BaseClass
include IncludedOtherModule
include IncludedModule
end
อาจมีพฤติกรรมแตกต่างไปจากเดิมอย่างสิ้นเชิง หากทั้งสองโมดูลมีเมธอดที่มีชื่อเหมือนกัน ลำดับที่นี่จะเป็นตัวกำหนดว่าโมดูลใดมีความสำคัญมากกว่า และ ที่เรียก super จะได้รับการแก้ไขเพื่อ โดยส่วนตัวแล้ว ฉันจะหลีกเลี่ยงการมีวิธีการที่ทับซ้อนกันเช่นนี้ให้มากที่สุด โดยเฉพาะอย่างยิ่งเพื่อหลีกเลี่ยงความกังวลเกี่ยวกับสิ่งต่าง ๆ เช่น ลำดับที่รวมโมดูลไว้
การใช้งานจริง
แม้ว่าจะเป็นการดีที่จะทราบความแตกต่างระหว่าง include และ prepend สำหรับโมดูล ฉันคิดว่าตัวอย่างในโลกแห่งความเป็นจริงจะช่วยแสดงให้เห็นว่าเมื่อใดที่คุณอาจเลือกอันใดอันหนึ่งมากกว่าอีกอันหนึ่ง กรณีการใช้งานหลักของฉันสำหรับโมดูลเช่นนี้คือเครื่องยนต์ Rails
อาจเป็นหนึ่งในเครื่องยนต์ Rails ที่ได้รับความนิยมมากที่สุด สมมติว่าเราต้องการเปลี่ยนอัลกอริทึมไดเจสต์รหัสผ่านที่กำลังใช้อยู่ แต่ก่อนอื่น ขอปฏิเสธความรับผิดชอบอย่างรวดเร็ว:
การใช้โมดูลในแต่ละวันของฉันคือการปรับแต่งการทำงานของกลไก Rails ที่มีตรรกะทางธุรกิจเริ่มต้นของเรา เรากำลังเอาชนะพฤติกรรมของโค้ด เราควบคุม . แน่นอน คุณสามารถใช้วิธีเดียวกันกับ Ruby ชิ้นใดก็ได้ แต่ ฉันจะไม่แนะนำให้แทนที่โค้ดที่คุณไม่ได้ควบคุม (เช่น จากอัญมณีที่ดูแลโดยผู้อื่น) เนื่องจากการเปลี่ยนแปลงใดๆ ในโค้ดภายนอกนั้นอาจเข้ากันไม่ได้กับการเปลี่ยนแปลงของคุณ
สรุปรหัสผ่านของ Devise เกิดขึ้นที่นี่ในโมดูล Devise::Models::DatabaseAuthenticatable:
def password_digest(password)
Devise::Encryptor.digest(self.class, password)
end
# and also in the password check:
def valid_password?(password)
Devise::Encryptor.compare(self.class, encrypted_password, password)
end
Devise ให้คุณปรับแต่งอัลกอริทึมที่ใช้ที่นี่โดยสร้าง Devise::Encryptable::Encryptors ของคุณเอง ซึ่งเป็นวิธีที่ถูกต้อง อย่างไรก็ตาม เพื่อจุดประสงค์ในการสาธิต เราจะใช้โมดูล
# app/models/password_digest_module
module PasswordDigestModule
def password_digest(password)
# Devise's default bcrypt is better for passwords,
# using sha1 here just for demonstration
Digest::SHA1.hexdigest(password)
end
def valid_password?(password)
Devise.secure_compare(password_digest(password), self.encrypted_password)
end
end
begin
User.include(PasswordDigestModule)
# Pro-tip - because we are calling User here, ActiveRecord will
# try to read from the database when this class is loaded.
# This can cause commands like `rails db:create` to fail.
rescue ActiveRecord::NoDatabaseError, ActiveRecord::StatementInvalid
end
หากต้องการโหลดโมดูลนี้ คุณจะต้องเรียก Rails.application.eager_load! ในการพัฒนาหรือเพิ่มตัวเริ่มต้น Rails เพื่อโหลดไฟล์ เมื่อทดสอบแล้ว เราพบว่าใช้งานได้ตามที่คาดไว้:
> User.create!(email: "one@test.com", name: "Test", password: "TestPassword")
=> #<User id: 1, name: "Test", created_at: "2021-05-01 02:08:29", updated_at: "2021-05-01 02:08:29", posts_count: nil, email: "one@test.com">
> User.first.valid_password?("TestPassword")
=> true
> User.first.encrypted_password
=> "4203189099774a965101b90b74f1d842fc80bf91"
ในกรณีของเราที่นี่ ทั้ง include และ prepend จะได้ผลลัพธ์เหมือนกัน แต่ขอเพิ่มความซับซ้อน จะเกิดอะไรขึ้นหากโมเดลผู้ใช้ของเราใช้ password_salt . ของตัวเอง เมธอด แต่เราต้องการแทนที่ในเมธอดของโมดูลของเรา:
class User < ApplicationRecord
# Include default devise modules. Others available are:
# :confirmable, :lockable, :timeoutable, :trackable and :omniauthable
devise :database_authenticatable, :registerable,
:recoverable, :rememberable, :validatable
has_many :posts
def password_salt
# Terrible way to create a password salt,
# purely for demonstration purposes
Base64.encode64(email)[0..-4]
end
end
จากนั้น เราอัปเดตโมดูลของเราเพื่อใช้ password_salt . ของตัวเอง เมธอดเมื่อสร้างไดเจสต์รหัสผ่าน:
def password_digest(password)
# Devise's default bcrypt is better for passwords,
# using sha1 here just for demonstration
Digest::SHA1.hexdigest(password + "." + password_salt)
end
def password_salt
# an even worse way of generating a password salt
"salt"
end
ตอนนี้ include และ prepend จะมีพฤติกรรมแตกต่างออกไปเพราะว่าอันไหนที่เราใช้เป็นตัวกำหนด password_salt วิธีการ Ruby ดำเนินการ ด้วย prepend , โมดูลจะมีความสำคัญ และเราได้รับสิ่งนี้:
> User.last.password_digest("test")
=> "a94a8fe5ccb19ba61c4c0873d391e987982fbbd3.salt"
เปลี่ยนโมดูลเพื่อใช้ include จะหมายถึงการนำคลาส User ไปใช้แทน:
> User.last.password_digest("test")
=> "a94a8fe5ccb19ba61c4c0873d391e987982fbbd3.dHdvQHRlc3QuY2"
โดยทั่วไป ฉันไปถึงสำหรับ prepend อย่างแรกเลย เพราะเมื่อเขียนโมดูล ฉันพบว่ามันง่ายกว่าที่จะปฏิบัติต่อมันเหมือนคลาสย่อยมากกว่า และถือว่าวิธีการใดๆ ในโมดูลจะแทนที่เวอร์ชันของคลาส เห็นได้ชัดว่านี่ไม่ใช่สิ่งที่ต้องการเสมอไป ซึ่งเป็นเหตุผลที่ Ruby ให้ include . แก่เรา ตัวเลือก
ทายาท
เราได้เห็นวิธีที่ Ruby ติดตามบรรพบุรุษของคลาสเพื่อทราบลำดับความสำคัญเมื่อดำเนินการเมธอด ตลอดจนวิธีที่เราแทรกรายการลงในรายการนี้ผ่านโมดูล อย่างไรก็ตาม ในฐานะโปรแกรมเมอร์ การวนซ้ำใน ผู้สืบทอด ของคลาสทั้งหมดอาจเป็นประโยชน์ , ด้วย. นี่คือที่ที่ #descendants . ของ ActiveSupport เมธอดเข้ามา วิธีการนี้ค่อนข้างสั้นและทำซ้ำได้ง่ายนอก Rails หากจำเป็น:
class Class
def descendants
ObjectSpace.each_object(singleton_class).reject do |k|
k.singleton_class? || k == self
end
end
end
ObjectSpace เป็นส่วนที่น่าสนใจมากของ Ruby ที่เก็บข้อมูลเกี่ยวกับ Ruby Object ทุกตัวที่อยู่ในหน่วยความจำ เราจะไม่ลงรายละเอียดที่นี่ แต่ถ้าคุณมีคลาสที่กำหนดไว้ในแอปพลิเคชันของคุณ (และมันถูกโหลดแล้ว) คลาสนั้นจะปรากฏใน ObjectSpace ObjectSpace#each_object เมื่อส่งผ่านโมดูล ส่งคืนเฉพาะอ็อบเจ็กต์ที่ตรงกันหรือเป็นคลาสย่อยของโมดูล การบล็อกที่นี่ยังปฏิเสธระดับบนสุดด้วย (เช่น ถ้าเราเรียก Numeric.descendants เราไม่คาดหวัง Numeric ให้อยู่ในผลลัพธ์)
อย่ากังวลหากคุณไม่เข้าใจสิ่งที่เกิดขึ้นที่นี่ เนื่องจากอาจจำเป็นต้องอ่านเพิ่มเติมเกี่ยวกับ ObjectSpace จึงจะเข้าใจได้ สำหรับจุดประสงค์ของเรา ก็เพียงพอแล้วที่จะรู้ว่าวิธีนี้ใช้กับ Class และส่งคืนรายการของคลาสที่สืบทอด หรือคุณอาจคิดว่ามันเป็น "แผนภูมิต้นไม้" ของลูก หลาน ฯลฯ ในชั้นเรียนนั้น
การใช้งาน #descendants ในโลกแห่งความเป็นจริง
ใน RailsConf ปี 2018 Ryan Laughlin ได้บรรยายเรื่อง "การตรวจสุขภาพ" วิดีโอควรค่าแก่การชม แต่เราจะแยกแนวคิดหนึ่งออกมา ซึ่งก็คือการเรียกใช้แถวทั้งหมดในฐานข้อมูลของคุณเป็นระยะๆ และตรวจสอบว่าผ่านการตรวจสอบความถูกต้องของแบบจำลองของคุณหรือไม่ คุณอาจแปลกใจว่าฐานข้อมูลของคุณมีกี่แถวที่ไม่ผ่าน #valid? ทดสอบ
คำถามคือ เราจะนำการตรวจสอบนี้ไปใช้ได้อย่างไรโดยไม่ต้องดูแลรายการแบบจำลองด้วยตนเอง #descendants คือคำตอบ:
# Ensure all models are loaded (should not be necessary in production)
Rails.application.load! if Rails.env.development?
ApplicationRecord.descendants.each do |model_class|
# in the real world you'd want to send this off to background job(s)
model_class.all.each do |record|
if !record.valid?
HoneyBadger.notify("Invalid #{model.name} found with ID: #{record.id}")
end
end
end
ที่นี่ ApplicationRecord.descendants แสดงรายการของทุกรุ่นในแอปพลิเคชัน Rails มาตรฐาน ในลูปของเราแล้ว model เป็นคลาส (เช่น User หรือ Product ). การใช้งานที่นี่ค่อนข้างเรียบง่าย แต่ผลที่ได้คือสิ่งนี้จะทำซ้ำในทุกโมเดล (หรือแม่นยำกว่าทุกคลาสย่อยของ ApplicationRecord) และเรียก .valid? สำหรับทุกแถว
บทสรุป
สำหรับนักพัฒนา Rails ส่วนใหญ่ มักไม่ใช้โมดูล นี่เป็นเหตุผลที่ดี หากคุณเป็นเจ้าของรหัส มักจะมีวิธีที่ง่ายกว่าในการปรับแต่งลักษณะการทำงาน และหากคุณไม่ เป็นเจ้าของรหัส มีความเสี่ยงในการเปลี่ยนพฤติกรรมด้วยโมดูล อย่างไรก็ตาม พวกเขามีกรณีการใช้งานและเป็นข้อพิสูจน์ถึงความยืดหยุ่นของ Ruby ที่ไม่เพียงแต่เราสามารถเปลี่ยนคลาสจากไฟล์อื่นได้ เรายังมีตัวเลือกให้เลือกที่ ในสายบรรพบุรุษ โมดูลของเราจะปรากฏขึ้น
ActiveSupport เข้ามาเพื่อให้ตรงกันข้ามของ #ancestors กับ #descendants . วิธีนี้ไม่ค่อยได้ใช้เท่าที่ฉันเคยเห็นมา แต่เมื่อคุณรู้ว่ามีวิธีนี้แล้ว คุณก็จะพบว่าวิธีนี้มีประโยชน์มากขึ้นเรื่อยๆ โดยส่วนตัวแล้ว ฉันไม่ได้ใช้มันเพื่อตรวจสอบความถูกต้องของโมเดลเท่านั้น แต่ยังใช้สเปกเพื่อตรวจสอบว่าเรากำลังเพิ่ม attribute_alias อย่างถูกต้องหรือไม่ วิธีการสำหรับทุกรุ่นของเรา