我之前很多次使用default_scope,但是后来万分后悔这么做。
假设我们在模型Post中定义一个default_scope:
class Post
default_scope where(published:true).order("created_at desc")
end
defaut_scope会增加一些你可能不期望的行为
1.你不能重写default_scope。例如:默认会按照created_at排序。
> Post.limit(10)
Post Load (3.3ms) SELECT `posts`.* FROM `posts` WHERE `posts`.`published` = 1 ORDER BY created_at desc LIMIT 10
如果你想按照updated_at排序,你可能按照如下的方式进行:
> Post.order("updated_at desc").limit(10)
Post Load (17.3ms) SELECT `posts`.* FROM `posts` WHERE `posts`.`published` = 1 ORDER BY created_at desc, updated_at desc LIMIT 10
但实际上,它会同时按照created_at和updated_at排序,default_scope没有被重写,你需要unscope方法来禁用default_scope。
> Post.unscoped.order("updated_at desc").limit(10)
Post Load (1.9ms) SELECT `posts`.* FROM `posts` ORDER BY updated_at desc LIMIT 10
2.default_scope会影响Model的初始化
> Post.new
=> #<Post id: nil, title: nil, created_at: nil, updated_at: nil, user_id: nil, published: true>
很多开发者没有意识到这一点,误以为default_scope只会对查询起作用,实则不然。
因此,尽量不要使用default_scope,仅仅把它定义为一个scope就很好,调用scope即可。