1 什麼是 Active Record?
Active Record 是 MVC 中 M(模型)的一部分,模型是系統中負責表示資料和業務邏輯的層。Active Record 可協助您建立和使用 Ruby 物件,這些物件的屬性需要持久儲存到資料庫。
Active Record 和 Active Model 之間有何差異?可以使用不需要資料庫支援的 Ruby 物件來建立資料模型。Active Model 通常在 Rails 中用於此目的,使 Active Record 和 Active Model 都成為 MVC 中 M 的一部分,以及您自己的純 Ruby 物件。
術語「Active Record」也指一種軟體架構模式。Rails 中的 Active Record 是該模式的實作。它也是對稱為物件關聯對應系統的描述。以下章節說明這些術語。
1.1 Active Record 模式
Martin Fowler 在《企業應用程式架構模式》一書中將 Active Record 模式描述為「一個包裝資料庫表格中的列的物件,封裝資料庫存取,並將網域邏輯新增至該資料。」Active Record 物件同時攜帶資料和行為。Active Record 類別與底層資料庫的記錄結構非常接近。這樣使用者就可以輕鬆地從資料庫讀取和寫入資料,正如您在下面的範例中所看到的。
1.2 物件關聯對應
物件關聯對應 (通常稱為 ORM) 是一種將程式語言的豐富物件連接到關聯式資料庫管理系統 (RDBMS) 中表格的技術。在 Rails 應用程式中,這些是 Ruby 物件。透過使用 ORM,Ruby 物件的屬性以及物件之間的關係可以輕鬆地儲存到資料庫並從中擷取,而無需直接撰寫 SQL 陳述式。總體而言,ORM 最大程度地減少了您必須撰寫的資料庫存取程式碼量。
若要完全理解 Active Record,對關聯式資料庫管理系統 (RDBMS) 和結構化查詢語言 (SQL) 的基本知識會有所幫助。請參閱此 SQL 教學課程(或此 RDBMS 教學課程),或者如果您想了解更多資訊,請透過其他方式學習。
1.3 Active Record 作為 ORM 框架
Active Record 讓我們可以使用 Ruby 物件執行下列操作
- 表示模型及其資料。
- 表示模型之間的關聯。
- 透過相關模型表示繼承階層。
- 在模型持續保存到資料庫之前驗證模型。
- 以物件導向的方式執行資料庫操作。
2 Active Record 中的慣例優於配置
當使用其他程式語言或框架撰寫應用程式時,可能需要撰寫大量的設定程式碼。對於一般 ORM 框架來說尤其如此。但是,如果您遵循 Rails 採用的慣例,在建立 Active Record 模型時,您將幾乎不用或完全不用撰寫設定程式碼。
Rails 採用一個觀念,即如果您大多數時間都以相同的方式設定應用程式,那麼該方式應該是預設方式。只有在您無法遵循慣例的情況下,才需要明確設定。
若要利用 Active Record 中慣例優於配置的優勢,需要遵循一些命名和結構描述慣例。如果您需要,也可以覆寫命名慣例。
2.1 命名慣例
Active Record 使用此命名慣例來對應模型(由 Ruby 物件表示)和資料庫表格
Rails 將對您模型的類別名稱進行複數化,以尋找各自的資料庫表格。例如,名為 Book
的類別會對應到名為 books
的資料庫表格。Rails 的複數化機制非常強大,能夠複數化(和單數化)英語中的規則和不規則單字。這使用 Active Support pluralize 方法。
對於由兩個或多個單字組成的類別名稱,模型類別名稱將遵循 Ruby 慣例,使用 UpperCamelCase 名稱。在這種情況下,資料庫表格名稱將是一個 snake_case 名稱。例如
BookClub
是模型類別,單數形式,每個單字的首字母大寫。book_clubs
是對應的資料庫表格,複數形式,單字之間用底線分隔。
以下是一些模型類別名稱和相應表格名稱的更多範例
模型/類別 | 表格/結構描述 |
---|---|
Article |
articles |
LineItem |
line_items |
Product |
products |
Person |
people |
2.2 結構描述慣例
Active Record 也對資料庫表格中的欄名稱使用慣例,具體取決於這些欄的目的。
- 主鍵 - 依預設,Active Record 將使用名為
id
的整數欄作為表格的主鍵 (PostgreSQL、MySQL 和 MariaDB 為bigint
,SQLite 為integer
)。當使用Active Record 遷移來建立表格時,此欄將會自動建立。 - 外來鍵 - 這些欄位應按照
singularized_table_name_id
模式命名 (例如,order_id
、line_item_id
)。這些是當您在模型之間建立關聯時 Active Record 將會尋找的欄位。
還有一些可選的欄名稱,將會為 Active Record 執行個體新增額外功能
created_at
- 在首次建立記錄時,會自動設定為目前日期和時間。updated_at
- 在每次建立或更新記錄時,會自動設定為目前日期和時間。lock_version
- 將樂觀鎖定新增至模型。type
- 指定模型使用單一表格繼承。(association_name)_type
- 儲存多型關聯的類型。(table_name)_count
- 用於快取關聯中屬於物件的數量。例如,如果Article
有許多Comment
,則articles
表格中的comments_count
欄將會快取每個文章的現有評論數。
雖然這些欄名稱是可選的,但它們由 Active Record 保留。在命名表格的欄時,請避開保留的關鍵字。例如,type
是一個保留的關鍵字,用於指定使用單一表格繼承 (STI) 的表格。如果您不使用 STI,請使用不同的單字來準確描述您要建立模型的資料。
3 建立 Active Record 模型
當產生 Rails 應用程式時,將會在 app/models/application_record.rb
中建立一個抽象的 ApplicationRecord
類別。ApplicationRecord
類別繼承自 ActiveRecord::Base
,它將常規 Ruby 類別轉換為 Active Record 模型。
ApplicationRecord
是您應用程式中所有 Active Record 模型的基本類別。若要建立新的模型,請子類別化 ApplicationRecord
類別,這樣就可以了
class Book < ApplicationRecord
end
這將會建立一個 Book
模型,對應資料庫中的 books
表格,其中表格的每個欄位都會對應到 Book
類別的屬性。Book
的一個實例可以代表 books
表格中的一列。可以使用像這樣的 SQL 語句來建立具有 id
、title
和 author
欄位的 books
表格
CREATE TABLE books (
id int(11) NOT NULL auto_increment,
title varchar(255),
author varchar(255),
PRIMARY KEY (id)
);
然而,這並不是在 Rails 中通常的做法。Rails 中的資料庫表格通常是使用 Active Record 遷移 而不是原始 SQL 來建立的。上述 books
表格的遷移可以像這樣產生
$ bin/rails generate migration CreateBooks title:string author:string
並產生這樣的結果
# Note:
# The `id` column, as the primary key, is automatically created by convention.
# Columns `created_at` and `updated_at` are added by `t.timestamps`.
# db/migrate/20240220143807_create_books.rb
class CreateBooks < ActiveRecord::Migration[8.0]
def change
create_table :books do |t|
t.string :title
t.string :author
t.timestamps
end
end
end
該遷移會建立欄位 id
、title
、author
、created_at
和 updated_at
。此表格的每一列都可以用具有相同屬性的 Book
類別的實例來表示:id
、title
、author
、created_at
和 updated_at
。您可以像這樣存取書本的屬性
irb> book = Book.new
=> #<Book:0x00007fbdf5e9a038 id: nil, title: nil, author: nil, created_at: nil, updated_at: nil>
irb> book.title = "The Hobbit"
=> "The Hobbit"
irb> book.title
=> "The Hobbit"
您可以使用命令 bin/rails generate model Book title:string author:string
來產生 Active Record 模型類別以及相符的遷移。這會建立檔案 app/models/book.rb
、db/migrate/20240220143807_create_books.rb
以及其他幾個用於測試的檔案。
3.1 建立命名空間模型
Active Record 模型預設會放置在 app/models
目錄下。但是您可能希望將類似的模型放在它們自己的資料夾和命名空間下,來組織您的模型。例如,將 order.rb
和 review.rb
放在 app/models/book
下,分別使用 Book::Order
和 Book::Review
類別名稱。您可以使用 Active Record 建立命名空間模型。
在 Book
模組尚未存在的情況下,generate
命令會像這樣建立所有內容
$ bin/rails generate model Book::Order
invoke active_record
create db/migrate/20240306194227_create_book_orders.rb
create app/models/book/order.rb
create app/models/book.rb
invoke test_unit
create test/models/book/order_test.rb
create test/fixtures/book/orders.yml
如果 Book
模組已經存在,系統會要求您解決衝突
$ bin/rails generate model Book::Order
invoke active_record
create db/migrate/20240305140356_create_book_orders.rb
create app/models/book/order.rb
conflict app/models/book.rb
Overwrite /Users/bhumi/Code/rails_guides/app/models/book.rb? (enter "h" for help) [Ynaqdhm]
在命名空間模型產生成功後,Book
和 Order
類別看起來會像這樣
# app/models/book.rb
module Book
def self.table_name_prefix
"book_"
end
end
# app/models/book/order.rb
class Book::Order < ApplicationRecord
end
在 Book
中設定 table_name_prefix 將允許 Order
模型的資料庫表格命名為 book_orders
,而不是單純的 orders
。
另一種可能性是您已經有一個想要保留在 app/models
中的 Book
模型。在這種情況下,您可以選擇 n
在 generate
命令期間不覆寫 book.rb
。
這仍然允許 Book::Order
類別使用命名空間的表格名稱,而無需使用 table_name_prefix
# app/models/book.rb
class Book < ApplicationRecord
# existing code
end
Book::Order.table_name
# => "book_orders"
4 覆寫命名慣例
如果您需要遵循不同的命名慣例,或是需要在具有舊版資料庫的情況下使用您的 Rails 應用程式該怎麼辦?沒問題,您可以輕鬆覆寫預設的慣例。
由於 ApplicationRecord
繼承自 ActiveRecord::Base
,因此您的應用程式模型會有很多可用的實用方法。例如,您可以使用 ActiveRecord::Base.table_name=
方法來自訂應該使用的表格名稱
class Book < ApplicationRecord
self.table_name = "my_books"
end
如果您這樣做,您必須使用測試定義中的 set_fixture_class
方法,手動定義託管 fixture (my_books.yml
) 的類別名稱
# test/models/book_test.rb
class BookTest < ActiveSupport::TestCase
set_fixture_class my_books: Book
fixtures :my_books
# ...
end
也可以使用 ActiveRecord::Base.primary_key=
方法來覆寫應該用作表格主鍵的欄位
class Book < ApplicationRecord
self.primary_key = "book_id"
end
Active Record 不建議使用名為 id
的非主鍵欄位。 使用名為 id
且不是單欄主鍵的欄位會使存取欄位值變得複雜。應用程式必須使用 id_value
別名屬性來存取非 PK id
欄位的值。
如果您嘗試建立一個名為 id
且不是主鍵的欄位,Rails 會在遷移期間拋出錯誤,例如:you can't redefine the primary key column 'id' on 'my_books'.
To define a custom primary key, pass { id: false } to create_table.
5 CRUD:讀取和寫入資料
CRUD 是用於操作資料的四個動詞的縮寫:Create(建立)、Read(讀取)、Update(更新)和 Delete(刪除)。Active Record 會自動建立方法,讓您可以讀取和操作儲存在應用程式資料庫表格中的資料。
Active Record 使用這些高階方法抽象化資料庫存取細節,使其可以無縫執行 CRUD 操作。請注意,所有這些方便的方法都會產生在基礎資料庫上執行的 SQL 語句。
以下範例顯示一些 CRUD 方法以及產生的 SQL 語句。
5.1 建立
Active Record 物件可以從雜湊、區塊建立,或者在建立後手動設定其屬性。new
方法會傳回一個新的、未持久化的物件,而 create
會將物件儲存到資料庫並傳回該物件。
例如,假設有一個具有 title
和 author
屬性的 Book
模型,則 create
方法呼叫會建立一個物件並將新的記錄儲存到資料庫中
book = Book.create(title: "The Lord of the Rings", author: "J.R.R. Tolkien")
# Note that the `id` is assigned as this record is committed to the database.
book.inspect
# => "#<Book id: 106, title: \"The Lord of the Rings\", author: \"J.R.R. Tolkien\", created_at: \"2024-03-04 19:15:58.033967000 +0000\", updated_at: \"2024-03-04 19:15:58.033967000 +0000\">"
而 new
方法會實例化一個物件,但不會將其儲存到資料庫中
book = Book.new
book.title = "The Hobbit"
book.author = "J.R.R. Tolkien"
# Note that the `id` is not set for this object.
book.inspect
# => "#<Book id: nil, title: \"The Hobbit\", author: \"J.R.R. Tolkien\", created_at: nil, updated_at: nil>"
# The above `book` is not yet saved to the database.
book.save
book.id # => 107
# Now the `book` record is committed to the database and has an `id`.
最後,如果提供了區塊,create
和 new
都會將新的物件產生到該區塊進行初始化,而只有 create
會將產生的物件持久化到資料庫中
book = Book.new do |b|
b.title = "Metaprogramming Ruby 2"
b.author = "Paolo Perrotta"
end
book.save
book.save
和 Book.create
產生的 SQL 語句看起來像這樣
/* Note that `created_at` and `updated_at` are automatically set. */
INSERT INTO "books" ("title", "author", "created_at", "updated_at") VALUES (?, ?, ?, ?) RETURNING "id" [["title", "Metaprogramming Ruby 2"], ["author", "Paolo Perrotta"], ["created_at", "2024-02-22 20:01:18.469952"], ["updated_at", "2024-02-22 20:01:18.469952"]]
5.2 讀取
Active Record 提供了豐富的 API 來存取資料庫中的資料。您可以查詢單一記錄或多個記錄、依任何屬性篩選它們、排序它們、分組它們、選取特定欄位,以及執行任何您可以使用 SQL 執行的操作。
# Return a collection with all books.
books = Book.all
# Return a single book.
first_book = Book.first
last_book = Book.last
book = Book.take
以上會產生以下 SQL
-- Book.all
SELECT "books".* FROM "books"
-- Book.first
SELECT "books".* FROM "books" ORDER BY "books"."id" ASC LIMIT ? [["LIMIT", 1]]
-- Book.last
SELECT "books".* FROM "books" ORDER BY "books"."id" DESC LIMIT ? [["LIMIT", 1]]
-- Book.take
SELECT "books".* FROM "books" LIMIT ? [["LIMIT", 1]]
我們也可以使用 find_by
和 where
找到特定的書籍。find_by
會傳回單一記錄,而 where
會傳回記錄清單
# Returns the first book with a given title or `nil` if no book is found.
book = Book.find_by(title: "Metaprogramming Ruby 2")
# Alternative to Book.find_by(id: 42). Will throw an exception if no matching book is found.
book = Book.find(42)
以上會產生此 SQL
-- Book.find_by(title: "Metaprogramming Ruby 2")
SELECT "books".* FROM "books" WHERE "books"."title" = ? LIMIT ? [["title", "Metaprogramming Ruby 2"], ["LIMIT", 1]]
-- Book.find(42)
SELECT "books".* FROM "books" WHERE "books"."id" = ? LIMIT ? [["id", 42], ["LIMIT", 1]]
# Find all books by a given author, sort by created_at in reverse chronological order.
Book.where(author: "Douglas Adams").order(created_at: :desc)
產生此 SQL
SELECT "books".* FROM "books" WHERE "books"."author" = ? ORDER BY "books"."created_at" DESC [["author", "Douglas Adams"]]
還有許多 Active Record 方法可以用來讀取和查詢記錄。您可以在 Active Record 查詢 指南中瞭解更多資訊。
5.3 更新
一旦擷取了 Active Record 物件,就可以修改其屬性,並將其儲存到資料庫中。
book = Book.find_by(title: "The Lord of the Rings")
book.title = "The Lord of the Rings: The Fellowship of the Ring"
book.save
一種簡寫方法是使用雜湊將屬性名稱對應到所需的值,如下所示
book = Book.find_by(title: "The Lord of the Rings")
book.update(title: "The Lord of the Rings: The Fellowship of the Ring")
update
會產生以下 SQL
/* Note that `updated_at` is automatically set. */
UPDATE "books" SET "title" = ?, "updated_at" = ? WHERE "books"."id" = ? [["title", "The Lord of the Rings: The Fellowship of the Ring"], ["updated_at", "2024-02-22 20:51:13.487064"], ["id", 104]]
這在一次更新多個屬性時很有用。與 create
類似,使用 update
會將更新的記錄提交到資料庫。
如果您想要批量更新多個記錄,而沒有回呼或驗證,則可以使用 update_all
直接更新資料庫
Book.update_all(status: "already own")
5.4 刪除
同樣地,一旦擷取了 Active Record 物件,就可以銷毀它,這會將其從資料庫中移除。
book = Book.find_by(title: "The Lord of the Rings")
book.destroy
destroy
會產生此 SQL
DELETE FROM "books" WHERE "books"."id" = ? [["id", 104]]
如果您想要批量刪除多個記錄,可以使用 destroy_by
或 destroy_all
方法
# Find and delete all books by Douglas Adams.
Book.destroy_by(author: "Douglas Adams")
# Delete all books.
Book.destroy_all
6 驗證
Active Record 允許您在模型寫入資料庫之前驗證模型的狀態。有幾種方法可以允許不同類型的驗證。例如,驗證屬性值是否為空、是否唯一、是否尚未在資料庫中、是否遵循特定格式等等。
像 save
、create
和 update
這樣的方法會在將模型持久化到資料庫之前驗證模型。如果模型無效,則不會執行任何資料庫操作。在這種情況下,save
和 update
方法會傳回 false
。create
方法仍然會傳回物件,可以檢查該物件是否有錯誤。所有這些方法都有一個驚嘆號對應方法(也就是 save!
、create!
和 update!
),這些方法更嚴格,因為它們會在驗證失敗時引發 ActiveRecord::RecordInvalid
例外。以下是一個快速範例來說明
class User < ApplicationRecord
validates :name, presence: true
end
irb> user = User.new
irb> user.save
=> false
irb> user.save!
ActiveRecord::RecordInvalid: Validation failed: Name can't be blank
create
方法始終會傳回模型,無論其有效性為何。然後,您可以檢查此模型是否有任何錯誤。
irb> user = User.create
=> #<User:0x000000013e8b5008 id: nil, name: nil>
irb> user.errors.full_messages
=> ["Name can't be blank"]
您可以在 Active Record 驗證指南中瞭解更多關於驗證的資訊。
7 回呼
Active Record 回呼允許您將程式碼附加到模型生命週期中的特定事件。這使您可以透過在這些事件發生時執行程式碼,來將行為新增至模型,例如在建立新記錄、更新它、銷毀它等等時。
class User < ApplicationRecord
after_create :log_new_user
private
def log_new_user
puts "A new user was registered"
end
end
irb> @user = User.create
A new user was registered
您可以在 Active Record 回呼指南中瞭解更多關於回呼的資訊。
8 遷移
Rails 提供了一種方便的方法,可以透過遷移來管理資料庫結構描述的變更。遷移是以特定領域語言編寫,並儲存在檔案中,這些檔案會針對 Active Record 支援的任何資料庫執行。
以下是一個建立名為 publications
的新表格的遷移
class CreatePublications < ActiveRecord::Migration[8.0]
def change
create_table :publications do |t|
t.string :title
t.text :description
t.references :publication_type
t.references :publisher, polymorphic: true
t.boolean :single_issue
t.timestamps
end
end
end
請注意,上述程式碼與資料庫無關:它將在 MySQL、MariaDB、PostgreSQL、SQLite 和其他資料庫中執行。
Rails 會追蹤哪些遷移已提交到資料庫,並將它們儲存在該資料庫中名為 schema_migrations
的相鄰表格中。
若要執行遷移並建立表格,您需要執行 bin/rails db:migrate
,若要回滾並刪除表格,則執行 bin/rails db:rollback
。
您可以在 Active Record 遷移指南中瞭解更多關於遷移的資訊。
9 關聯
Active Record 關聯允許您定義模型之間的關係。關聯可以用來描述一對一、一對多和多對多的關係。例如,「作者有多本書」這樣的關係可以定義如下
class Author < ApplicationRecord
has_many :books
end
Author
類別現在有方法可以新增和移除作者的書籍,以及更多功能。
您可以在 Active Record 關聯指南中瞭解更多關於關聯的資訊。