Rails N+1問題 with joins, includes, preload and eager_load

資料庫的操作是影響網站效能的重要因素,而在存取資料時幾乎一定會遇到 N+1 問題。本篇文章將簡要探討 includes 的使用時機,以及與其相似的 preload 和 eager_load 方法。此外,我們還會介紹在處理多表格操作時常用的 joins。 1. 造成 N+1 問題的原因 我們先想像一個情境,你正在開發商家的訂單系統,其中有兩個 Model,分別是Customer與Order,Customer記載顧客資料,並且每個顧客擁有多筆訂單Order # customer.rb class Customer < ApplicationRecord has_many :orders end # order.rb class Order < ApplicationRecord belongs_to :customer end 現在我們想要呈現4位顧客的個人資訊以及他們各自的所有訂單,你可能會這樣寫: Customer.limit(4).each{|customer| puts customer.orders} Rails ORM 產生的 SQL 指令如下: SELECT 'customers'.* FROM 'customers' LIMIT 4 SELECT 'orders'.* FROM 'orders' WHERE 'orders'.'customer_id'=1 SELECT 'orders'.* FROM 'orders' WHERE 'orders'.'customer_id'=2 SELECT 'orders'.* FROM 'orders' WHERE 'orders'.'customer_id'=3 SELECT 'orders'.* FROM 'orders' WHERE 'orders'....

June 8, 2023 · 291 words · HSIAO, YI-HUSAN

Rails 基礎 ORM 的 SQL 語法

儘管Rails開發者可以透過 ActiveRecord 簡單地操作資料庫資源,但仍然需要了解基本的 SQL 語法。以下是常見 ORM 指令的翻譯,可以幫助你真正理解背後的運作原理。 較複雜的資料庫關係語法(例如:includes、join)將不在條列於本文中,我們將在另一篇文章中探討這些內容。 Create create: 使用 create 創建資料,或透過 new 產生物件之後再使用 save,以上狀況所產生的SQL語法如下。 Book.create(name: "學習Rails", author: "Eddie", intro: "學習Rails好幫手", price: 100) or Book.new(name: "學習Rails", author: "Eddie", intro: "學習Rails好幫手", price: 100) Book.save INSERT INTO books(name, author, intro, price, updated_at) VALUES ('學習Rails', 'Eddie', '學習Rails好幫手', 100, '2023...') Read all: Book.all SELECT * FROM books select or pluck: select 和 pluck 最終得到的資料形式不同,這裡不多做討論,只要知道Rails進一步的處理是在從資料庫抓取資料後,兩者前期的SQL語法沒有不同 Book.select("name") Book.pluck("name") SELECT name FROM books limit or offset: Book....

June 4, 2023 · 510 words · HSIAO, YI-HUSAN

Rails ORM 常見的CRUD指令

CRUD = Create Read Update Delete是網站開發中的基本,在此筆記常用到的Rails CRUD指令。 Create new:產生新資料,但不會存檔 create:產生新資料,會直接存檔 create!:與create相同,但過程中發生錯誤時會報錯 new_book = Book.new(name: "老人與海", price: 300) new_book.save Book.create(name: "Les Misérables", author: "Victor Hugo") Read all:一次抓取所有資料 limit():抓去特定數量資料 first & last:找到該資料類型中的第一筆 & 最後一筆資料 find_by(id: 1):找到第一筆符合條件的資料,找不到會回傳nil find_by!(id: 1):與find_by相同,但是找不到資料會報錯 find(1):依據id找尋資料,找不到資料會報錯 select('name'):只抓出資料表中的特定欄位(資料表欄位太多時可以節省記憶體空間) where():以陣列的形式回傳找到的多筆符合條件的資料,找不到回傳空矩陣 find_by_sql():使用sql語法查詢(較少使用) order('price AESC'):依照遞增或遞減順序抓取所有資料 order(price: :aesc):order的不同寫法 find_each:batch find寫法,預設先抓出1000筆資料,當資料量太多時使用 Book.all Book.limit(5) // 抓取前五筆資料 Book.offset(5).limit(5) // 抓取下五筆資料 Book.first Book.last Book.find(id: 10) Book.find_by(id: 10, author: "Victor Hugo") // 依據多個條件尋找 and Book.find_by!(id: 9999) //報錯: in `find_by!': Couldn't find Book (ActiveRecord::RecordNotFound) Book....

April 4, 2023 · 186 words · HSIAO, YI-HUSAN