數(shù)據(jù)庫(kù)高級(jí)使用

1盒件、在Model使用sql語(yǔ)句

before_destroy :clean_data

def clean_data
  teacher_id = self.id
  sql = "delete * from train_teacher where nurse_info_id = #{teacher_id}"
  #這個(gè)方法是私有方法褪迟,必須在Model中使用
  ActiveRecord::Base.connection.exec(sql)
end

2厕妖、數(shù)據(jù)庫(kù)的優(yōu)化
一. N+1 queries 問(wèn)題

# model
class User < ActieRecord::Base
  has_one :car
end

class Car < ActiveRecord::Base
  belongs_to :user
end

# your controller
def index
  @users = User.page(params[:page])
end

# view
<% @users.each do |user| %>
 <%= user.car.name %>
<% end %>

當(dāng)我們?cè)赩iew中使用user.car.name的時(shí)候塘偎,會(huì)導(dǎo)致N+1 queries(多次查詢(xún)數(shù)據(jù)庫(kù))剧腻,假設(shè)User有10筆,這程式會(huì)產(chǎn)生出11筆Queries隔嫡,一筆是查
User甸怕,另外10筆是一筆一筆去查Car甘穿,嚴(yán)重拖慢效能腮恩。

SELECT * FROM `users` LIMIT 10 OFFSET 0
SELECT * FROM `cars` WHERE (`cars`.`user_id` = 1)
SELECT * FROM `cars` WHERE (`cars`.`user_id` = 2)
SELECT * FROM `cars` WHERE (`cars`.`user_id` = 3)
...
...
...
SELECT * FROM `cars` WHERE (`cars`.`user_id` = 10)

解決方法,加上includes:

# your controller
def index
  @users = User.includes(:car).page(params[:page])
end

如此SQL query就只有兩個(gè)温兼,只用一個(gè)就查出所有Cars資料秸滴。

SELECT * FROM `users` LIMIT 10 OFFSET 0
SELECT * FROM `cars` WHERE (`cars`.`user_id` IN('1','2','3','4','5','6','7','8',' 9','10'))

二. 索引
需要加索引的列:

  • 外鍵(Foreign Key)
  • 需要排序的列
  • 被查詢(xún)的列
  • 會(huì)被group的列

三. 使用select
ActiveRecord會(huì)將數(shù)據(jù)庫(kù)中的所有列取出來(lái),浪費(fèi)內(nèi)存募判。

#只取出id,name,description這些字段
Event.select(:id, :name, :description).limit(10)

四. 使用計(jì)數(shù)器
在ActiveRecord中經(jīng)常要計(jì)算has_many的Model有多少關(guān)聯(lián)的數(shù)據(jù)荡含。

<% @topics.each do |topic| %>
  主題:<%= topic.subject %>
  回復(fù)數(shù):<%= topic.posts.size %>
<% end %>

這個(gè)時(shí)候會(huì)產(chǎn)生很多SQL查詢(xún)。

SELECT * FROM `posts` LIMIT 5 OFFSET 0
SELECT count(*) AS count_all FROM `posts` WHERE (`posts`.topic_id = 1 )
SELECT count(*) AS count_all FROM `posts` WHERE (`posts`.topic_id = 2 )
SELECT count(*) AS count_all FROM `posts` WHERE (`posts`.topic_id = 3 )
SELECT count(*) AS count_all FROM `posts` WHERE (`posts`.topic_id = 4 )
SELECT count(*) AS count_all FROM `posts` WHERE (`posts`.topic_id = 5 )

使用counter cache會(huì)把這個(gè)數(shù)字存進(jìn)數(shù)據(jù)庫(kù)届垫。必須在Topic這個(gè)Model增加一個(gè)字段释液。

rails g migration add_posts_count_to_topic

編輯Migration文件

class AddPostsCountToTopic < ActiveRecord::Migration
  def change  
    add_column :topics, :posts_count, :integer, :default => 0
    
    Topic.pluck(:id).each do |i|
      Topic.reset_counters(i, :posts) # 全部重算一次
    end
  end
end

編輯Model文件,加入counter_cache => true

class Topic < ActiveRecord::Base
  has_many :posts
end

class Posts < ActiveRecord::Base
  belongs_to :topic, :counter_cache => true
end

這樣topic.posts.size會(huì)自動(dòng)轉(zhuǎn)化為topic.posts_count
五. 查詢(xún)優(yōu)化
如果要查出所有的數(shù)據(jù)装处,不要用all误债,all會(huì)一次把所有的數(shù)據(jù)取到內(nèi)存中去。


六. 逆正規(guī)化

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末妄迁,一起剝皮案震驚了整個(gè)濱河市寝蹈,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌登淘,老刑警劉巖箫老,帶你破解...
    沈念sama閱讀 216,544評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異黔州,居然都是意外死亡耍鬓,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門(mén)流妻,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)界斜,“玉大人,你說(shuō)我怎么就攤上這事合冀「鬓保” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,764評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)峭判。 經(jīng)常有香客問(wèn)我开缎,道長(zhǎng),這世上最難降的妖魔是什么林螃? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,193評(píng)論 1 292
  • 正文 為了忘掉前任奕删,我火速辦了婚禮,結(jié)果婚禮上疗认,老公的妹妹穿的比我還像新娘完残。我一直安慰自己,他們只是感情好横漏,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,216評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布谨设。 她就那樣靜靜地躺著,像睡著了一般缎浇。 火紅的嫁衣襯著肌膚如雪扎拣。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,182評(píng)論 1 299
  • 那天素跺,我揣著相機(jī)與錄音二蓝,去河邊找鬼。 笑死指厌,一個(gè)胖子當(dāng)著我的面吹牛刊愚,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播踩验,決...
    沈念sama閱讀 40,063評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼鸥诽,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了晰甚?” 一聲冷哼從身側(cè)響起衙传,我...
    開(kāi)封第一講書(shū)人閱讀 38,917評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎厕九,沒(méi)想到半個(gè)月后蓖捶,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,329評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡扁远,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,543評(píng)論 2 332
  • 正文 我和宋清朗相戀三年俊鱼,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片畅买。...
    茶點(diǎn)故事閱讀 39,722評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡并闲,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出谷羞,到底是詐尸還是另有隱情帝火,我是刑警寧澤溜徙,帶...
    沈念sama閱讀 35,425評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站犀填,受9級(jí)特大地震影響蠢壹,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜九巡,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,019評(píng)論 3 326
  • 文/蒙蒙 一图贸、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧冕广,春花似錦疏日、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,671評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至神凑,卻和暖如春净神,著一層夾襖步出監(jiān)牢的瞬間何吝,已是汗流浹背溉委。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,825評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留爱榕,地道東北人瓣喊。 一個(gè)月前我還...
    沈念sama閱讀 47,729評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像黔酥,于是被迫代替她去往敵國(guó)和親藻三。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,614評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容