在開(kāi)發(fā)web項(xiàng)目中一個(gè)我們很常見(jiàn)的場(chǎng)景就是绳军,我們需要去跟蹤模型記錄的改變惩坑,要知道記錄是從什么值變?yōu)槭裁粗邓氡谩ails通過(guò)提供了ActiveModel::Dirty中的一系列方法吁峻,來(lái)很好的幫助我們解決這些問(wèn)題卖漫,讓我們來(lái)看看有哪些好用的方法:
首先假設(shè)我們有 Card這個(gè)模型其中有name這個(gè)屬性
1. .changed? and <attribute>_changed?
這兩個(gè)方法是用于判斷記錄或?qū)傩允欠癖恍薷模?/p>
card = Card.first
card.name # => "克洛澤"
card.changed? # => false
card.name_changed? # => false
card.name = "Klose"
card.changed? # => true
card.name_changed? #=> true
如上面代碼所示宣旱,使用ActiveModel::Dirty提供的這兩個(gè)方法可以很好的幫助我們仅父,讓我們知道記錄或者屬性是否被修改了。
2. <attribute>_was
如果我們想知道一個(gè)屬性被修改前的值的話浑吟,ActiveModel::Dirty也提供了相應(yīng)的方法笙纤,那就是:
card.name = 'Klose'
card.name_was # => '克洛澤'
3. changes
上面我們已經(jīng)知道了如何得到一個(gè)屬性被修改之前的值,那么使用changes方法组力,我們將會(huì)得到省容,所有被修改的屬性及其遷移值的哈希
card.name = 'Klose'
card.changes # => {"name"=>["克洛澤", "Klose"]}
changes 返回的是以被修改的屬性的名字作為鍵,以原有值和當(dāng)前值的兩個(gè)元素的數(shù)組為值的哈希燎字。
以上是 Rails的ActiveModel::Dirty中提供的一組跟蹤記錄修改的方法腥椒,不過(guò)其中還有有一個(gè)非常重要的方法沒(méi)有介紹。
4. previous_changes
在上面的例子中候衍,當(dāng)我們把記錄持久化之后:
card.name = "Klose"
card.changed? # => true
card.save
card.changed? # => false
card.name_changed? #=> false
如你所見(jiàn)笼蛛,changed?和其他的方法并不能跟蹤已經(jīng)持久化的記錄,不過(guò)ActiveModel::Dirty中還是提供了
previous_changes這個(gè)方法脱柱,它能夠讓我們?cè)谟涗浺呀?jīng)被持久化后還能知道記錄是否被修改伐弹,以及修改之前的值:
card.name = "Klose"
card.save
card.previous_changes # => {"name"=>["克洛澤", "Klose"], "updated_at"=>[Sat, 09 Apr 2016 14:56:39 CST +08:00, Sat, 09 Apr 2016 15:08:03 CST +08:00]}
previous_changes可以返回,記錄被持久后榨为,哪些值被修改了以及修改前后的值惨好,還有修改時(shí)間,如果記錄沒(méi)有被改變的話随闺,返回的是一個(gè)空的Hash日川。
這個(gè)方法的一個(gè)非常常見(jiàn)的使用場(chǎng)景就是,使用after_update回調(diào)來(lái)進(jìn)行矩乐,對(duì)記錄修改后的通知功能龄句。
class Card < ActiveRecord
after_update :notify_if_name_changed
private
def notify_if_name_changed
notify.call if self.previous_changes['name']
end
end
當(dāng)然其實(shí)這里還有另外一個(gè)方法回论,去實(shí)現(xiàn)記錄修改后的通知功能,而且不使用previous_changes分歇。
class Card < ActiveRecord
around_update :notify_if_name_changed
private
def notify_if_name_changed
changed = self.name_changed?
yield
notify.call if changed
end
end
這個(gè)解決辦法中傀蓉,我們使用了環(huán)繞回調(diào)方法, 該方法是被用yield分隔的兩部分
- 在yield之前的代碼或者發(fā)生修改之前執(zhí)行
- 而在yield之后是修改已經(jīng)發(fā)生后執(zhí)行的
那么yield其實(shí)就是觸發(fā)這個(gè)修改動(dòng)作的標(biāo)識(shí)职抡,并且因?yàn)橥谝粋€(gè)方法中被分隔的兩部分的變量是在同一作用于下的葬燎,因此我們就可以使用它實(shí)現(xiàn)改成的功能了。
以上的幾個(gè)方法是跟蹤修改的實(shí)際場(chǎng)景中非常有用的缚甩,希望本文能夠?qū)δ阌袔椭?/p>