大概翻了一下 Effective Ruby (中文、原文),覺得其中第四個 Be Aware That Constants Are Mutable 有點意思,記錄下來。
在 Ruby 裡面大寫開頭的叫做常數,Ruby 開發者可能知道這個常數是可以被事後修改的,雖然會有警告啦,但是還是被修改到了,那有沒有辦法可以真正無法被修改呢? 原來要用 freeze
,而且還需要點技巧。
讓我們看一下代碼:
X = 1
X = 2
(irb):17: warning: already initialized constant X 只是警告而已
# X 變成 2 了 :(
直接 X.freeze
是沒有用的…
X.freeze
X = 3
(irb):20: warning: already initialized constant X
# X 還是變成 3 了 :(
解法:要用一個 module 包起來,然後 freeze 這個 module
module Y
X = 1
end
Y.freeze
Y::X = 2
# RuntimeError: can't modify frozen Module 丟出錯誤例外,不能修改!
接下來看看使用容器的情況:
class A
B = ["a", "b", "c"].freeze
def self.mutate
B[0] << "x"
end
end
A::B << "d"
# RuntimeError: can't modify frozen Array 丟出錯誤例外,不能修改!
A.mutate
A::B # 被修改成 ["ax", "b", "c"] 了,失敗 :(
Ruby 開發者大概已經知道只 freeze 一個容器,只能防止新增和刪除元素,不能阻止個別的元素被直接修改。所以解法是每個元素都要再 freeze 一次:
class A
B = ["a", "b", "c"].map!(&:freeze).freeze
def self.mutate
B[0] << "x"
end
end
A::B << "d"
# RuntimeError: can't modify frozen Array 丟出錯誤例外,不能修改!
A.mutate
# RuntimeError: can't modify frozen String 丟出錯誤例外,不能修改!
這樣就搞定啦。
應該是 be aware