教程原文http://guides.rubyonrails.org/active_record_querying.html
1、獲取數據
.獲取第一條、最後一條記錄
.通過id獲取記錄
.find all
.對一組數據進行相同操作
如果表記錄數比較大,這種方式比較耗資源,因爲它會一次載入整個表的數據。改用以下這種方式,它每次只載入1000行,然後逐步yield完整個表
自定義方式,find_each接受和find同樣的options
find_in_batches,和find_each相似,但它yield時傳遞的是model對象數組,而不是單個model對象
2、查詢條件
通過替換?來傳遞條件值,可避免SQL注入
symbol佔位條件
範圍條件 in(集合)
生成sql
如果要生成日期時間,再加上.to_time
params[:start_date].to_date.to_time,生成2007-12-01 00:00:00格式
有上數據庫會在以上條件中報錯,如Mysql會報查詢語句過長的錯誤,此時可以改成created_at > ? AND created_at < ?的形式
Hash條件
帶範圍條件
生成sql
集合條件
生成sql
3、查詢選項
排序
返回指定字段
限定和偏移Limit and Offset
Group分組
生成sql
Having
生成sql
只讀
更新時鎖定記錄
樂觀鎖Optimistic Locking
爲使用樂觀鎖,須在表裏建一個lock_version的字段,每次更新記錄時,ActiveRecord自動遞增lock_version的值,
備註:You must ensure that your database schema defaults the lock_version column to 0.
This behavior can be turned off by setting ActiveRecord::Base.lock_optimistically = false.
指定樂觀鎖字段名
悲觀鎖Pessimistic Locking
悲觀鎖定由數據庫直接提供
Mysql執行返回
SQL (0.2ms) BEGIN Item Load (0.3ms) SELECT * FROM `items` LIMIT 1 FOR UPDATE Item Update (0.4ms) UPDATE `items` SET `updated_at` = '2009-02-07 18:05:56', `name` = 'Jones' WHERE `id` = 1 SQL (0.8ms) COMMIT
爲特定數據庫加入原始的lock聲明
爲Mysql的鎖定聲明爲共享模式,即鎖定時仍然可讀
Item.transaction do i = Item.find(1, :lock => "LOCK IN SHARE MODE") i.increment!(:views) end
4、關聯表
生成sql
使用Array、Hash、Named Associations關聯表
有如下model
爲關聯查詢結果設定條件
5、優化載入
以下代碼,需要執行1 + 10次sql
優化:
一次性載入post的所有分類和評論
載入category爲1的所有post和cooment及tag
6、動態查詢
7、find_by_sql
8、select_all
和find_by_sql類似,但不會用model實例化返回記錄,你會得到一個hash數組
9、判斷記錄是否存在
10、計算
1、獲取數據
.獲取第一條、最後一條記錄
- Model.first
- Model.first(options)
- Model.find(:first, options)
- Model.last
- Model.last(options)
- Model.find(:last, options)
.通過id獲取記錄
.find all
.對一組數據進行相同操作
如果表記錄數比較大,這種方式比較耗資源,因爲它會一次載入整個表的數據。改用以下這種方式,它每次只載入1000行,然後逐步yield完整個表
自定義方式,find_each接受和find同樣的options
- User.find_each(:batch_size => 5000, :start => 2000) do |user|
- NewsLetter.weekly_deliver(user)
- end
find_in_batches,和find_each相似,但它yield時傳遞的是model對象數組,而不是單個model對象
- Invoice.find_in_batches(:include => :invoice_lines) do |invoices|
- export.add_invoices(invoices)
- end
2、查詢條件
通過替換?來傳遞條件值,可避免SQL注入
symbol佔位條件
- Client.all(:conditions => ["created_at >= :start_date AND created_at <= :end_date", {:start_date => params[:start_date], :end_date => params[:end_date] }])
範圍條件 in(集合)
- Client.all(:conditions => ["created_at IN (?)", (params[:start_date].to_date)..(params[:end_date].to_date])
生成sql
- SELECT * FROM users WHERE (created_at IN ('2007-12-31','2008-01-01','2008-01-02','2008-01-03','2008-01-04','2008-01-05', '2008-01-06','2008-01-07','2008-01-08'))
如果要生成日期時間,再加上.to_time
params[:start_date].to_date.to_time,生成2007-12-01 00:00:00格式
有上數據庫會在以上條件中報錯,如Mysql會報查詢語句過長的錯誤,此時可以改成created_at > ? AND created_at < ?的形式
Hash條件
帶範圍條件
生成sql
- SELECT * FROM clients WHERE (clients.created_at BETWEEN '2008-12-21 00:00:00' AND '2008-12-22 00:00:00')
集合條件
生成sql
3、查詢選項
排序
- #單個排序
- Client.all(:order => "created_at ASC")
- #多個排序
- Client.all(:order => "orders_count ASC, created_at DESC")
返回指定字段
限定和偏移Limit and Offset
- Client.all(:limit => 5)
- #生成
- SELECT * FROM clients LIMIT 5
- Client.all(:limit => 5, :offset => 5)
- #生成
- SELECT * FROM clients LIMIT 5, 5
Group分組
生成sql
Having
生成sql
只讀
- client = Client.first(:readonly => true)
- client.locked = false
- client.save
- #對只讀對象進行保存將會觸發ActiveRecord::ReadOnlyRecord異常
更新時鎖定記錄
樂觀鎖Optimistic Locking
爲使用樂觀鎖,須在表裏建一個lock_version的字段,每次更新記錄時,ActiveRecord自動遞增lock_version的值,
- c1 = Client.find(1) c2 = Client.find(1) c1.name = "Michael" c1.save c2.name = "should fail" c2.save # Raises a ActiveRecord::StaleObjectError
備註:You must ensure that your database schema defaults the lock_version column to 0.
This behavior can be turned off by setting ActiveRecord::Base.lock_optimistically = false.
指定樂觀鎖字段名
悲觀鎖Pessimistic Locking
悲觀鎖定由數據庫直接提供
Mysql執行返回
SQL (0.2ms) BEGIN Item Load (0.3ms) SELECT * FROM `items` LIMIT 1 FOR UPDATE Item Update (0.4ms) UPDATE `items` SET `updated_at` = '2009-02-07 18:05:56', `name` = 'Jones' WHERE `id` = 1 SQL (0.8ms) COMMIT
爲特定數據庫加入原始的lock聲明
爲Mysql的鎖定聲明爲共享模式,即鎖定時仍然可讀
Item.transaction do i = Item.find(1, :lock => "LOCK IN SHARE MODE") i.increment!(:views) end
4、關聯表
生成sql
使用Array、Hash、Named Associations關聯表
有如下model
- class Category < ActiveRecord::Base
- has_many :posts
- end
- class Post < ActiveRecord::Base
- belongs_to :category
- has_many :comments
- has_many :tags
- end
- class Comments <ActiveRecord::Base
- belongs_to :post
- has_one :guest
- end
- class Guest < ActiveRecord::Base
- belongs_to :comment
- end
- #關聯一個關係
- Category.all :joins => :posts
- #關聯多個關係
- Post.all :joins => [:category, :comments]
- #嵌套關聯
- Category.all :joins => {:posts => [{:comments => :guest}, :tags]}
爲關聯查詢結果設定條件
- time_range = (Time.now.midnight - 1.day)..Time.now.midnight Client.all :joins => :orders, :conditions => {'orders.created_at' => time_ran
- #或者
- time_range = (Time.now.midnight - 1.day)..Time.now.midnight Client.all :joins => :orders, :conditions => {:orders => {:created_at => time_range}}
5、優化載入
以下代碼,需要執行1 + 10次sql
優化:
- clients = Client.all(:include => :address, :limit => 10)
- clients.each do |client|
- puts client.address.postcode
- end
一次性載入post的所有分類和評論
載入category爲1的所有post和cooment及tag
6、動態查詢
- Client.find_by_name("Ryan")
- Client.find_all_by_name("Ryan")
- #!方法,沒有記錄時拋出ActiveRecord::RecordNotFound異常
- Client.find_by_name!("Ryan")
- #查詢多個字段
- Client.find_by_name_and_locked("Ryan", true)
- #查詢不到時就創建並保存
- Client.find_or_create_by_name(params[:name])
- #查詢不到時創建一個實例,但不保存
- Client.find_or_initialize_by_name('Ryan')
7、find_by_sql
- Client.find_by_sql("SELECT * FROM clients INNER JOIN orders ON clients.id = orders.client_id ORDER clients.created_at desc")
8、select_all
和find_by_sql類似,但不會用model實例化返回記錄,你會得到一個hash數組
9、判斷記錄是否存在
- #通過id來查詢
- Client.exists?(1)
- Client.exists?(1, 2, 3)
- #or
- Client.exists?([1,2,3])
- #通過其他條件來查詢
- Client.exists?(:conditions => "first_name = 'Ryan'")
- #沒有參數時,則:表是空的 ? false : true
- Client.exists?
10、計算