我如何在ActiveRecord设置默认值?
我看到Pratik的一篇文章,描述了一段丑陋而复杂的代码:http://m.onkey.org/2007/7/24/how-to-set-default-values-in-your-model
class Item < ActiveRecord::Base
def initialize_with_defaults(attrs = nil, &block)
initialize_without_defaults(attrs) do
setter = lambda { |key, value| self.send("#{key.to_s}=", value) unless
!attrs.nil? && attrs.keys.map(&:to_s).include?(key.to_s) }
setter.call('scheduler_type', 'hotseat')
yield self if block_given?
end
end
alias_method_chain :initialize, :defaults
end
我在谷歌上看到了以下例子:
def initialize
super
self.status = ACTIVE unless self.status
end
and
def after_initialize
return unless new_record?
self.status = ACTIVE
end
我也见过有人把它放在迁移中,但我更愿意看到它在模型代码中定义。
是否有一个规范的方法来设置默认值的字段在ActiveRecord模型?
我遇到了问题after_initialize给出ActiveModel::MissingAttributeError错误时做复杂的发现:
eg:
@bottles = Bottle.includes(:supplier, :substance).where(search).order("suppliers.name ASC").paginate(:page => page_no)
where中的"search"是条件哈希值
所以我用这种方式重写了初始化:
def initialize
super
default_values
end
private
def default_values
self.date_received ||= Date.current
end
super调用是必要的,以确保在执行自定义代码(即:default_values)之前从ActiveRecord::Base正确初始化对象
每个可用的方法都有几个问题,但我认为定义一个after_initialize回调是正确的方法,原因如下:
default_scope will initialize values for new models, but then that will become the scope on which you find the model. If you just want to initialize some numbers to 0 then this is not what you want.
Defining defaults in your migration also works part of the time... As has already been mentioned this will not work when you just call Model.new.
Overriding initialize can work, but don't forget to call super!
Using a plugin like phusion's is getting a bit ridiculous. This is ruby, do we really need a plugin just to initialize some default values?
Overriding after_initialize is deprecated as of Rails 3. When I override after_initialize in rails 3.0.3 I get the following warning in the console:
弃用警告:Base#after_initialize已弃用,请使用Base。After_initialize:方法代替。(调用from /Users/me/myapp/app/models/my_model:15)
因此,我要说的是写一个after_initialize回调函数,它允许你在关联上设置默认属性,就像这样:
class Person < ActiveRecord::Base
has_one :address
after_initialize :init
def init
self.number ||= 0.0 #will set the default value only if it's nil
self.address ||= build_address #let's you set a default association
end
end
现在您只有一个地方可以寻找模型的初始化。我一直在用这个方法,直到有人提出更好的方法。
警告:
For boolean fields do:
self.bool_field = true if self.bool_field.nil?
See Paul Russell's comment on this answer for more details
If you're only selecting a subset of columns for a model (ie; using select in a query like Person.select(:firstname, :lastname).all) you will get a MissingAttributeError if your init method accesses a column that hasn't been included in the select clause. You can guard against this case like so:
self.number ||= 0.0 if self.has_attribute? :number
and for a boolean column...
self.bool_field = true if (self.has_attribute? :bool_value) && self.bool_field.nil?
Also note that the syntax is different prior to Rails 3.2 (see Cliff Darling's comment below)
一些简单的情况可以通过在数据库模式中定义默认值来处理,但这不能处理许多棘手的情况,包括其他模型的计算值和键。对于这些情况,我这样做:
after_initialize :defaults
def defaults
unless persisted?
self.extras||={}
self.other_stuff||="This stuff"
self.assoc = [OtherModel.find_by_name('special')]
end
end
我决定使用after_initialize,但我不希望它应用于只发现那些新的或创建的对象。我认为几乎令人震惊的是,这个明显的用例没有提供一个after_new回调,但我已经通过确认对象是否已经被持久化来表明它不是新的。
看过Brad Murray的回答后,如果条件被移动到回调请求,这就更加清晰了:
after_initialize :defaults, unless: :persisted?
# ":if => :new_record?" is equivalent in this context
def defaults
self.extras||={}
self.other_stuff||="This stuff"
self.assoc = [OtherModel.find_by_name('special')]
end
我强烈建议使用“default_value_for”gem: https://github.com/FooBarWidget/default_value_for
有一些棘手的场景需要重写initialize方法,这个gem就是这样做的。
例子:
你的db默认值是NULL,你的模型/ruby定义的默认值是“一些字符串”,但你实际上想要设置值为nil的原因:MyModel。新(my_attr: nil)
这里的大多数解决方案将无法将值设置为nil,而是将其设置为默认值。
好吧,所以你不用||=方法,而是切换到my_attr_changed?…
但是现在假设你的数据库默认值是"some string",你的模型/ruby定义的默认值是"some other string",但在某种情况下,你想将值设置为"some string"(数据库默认值):MyModel。新(my_attr:“some_string”)
这将导致my_attr_changed?为false是因为该值与db默认值匹配,而db默认值将触发ruby定义的默认代码,并将该值设置为“其他字符串”——同样,这不是您想要的。
由于这些原因,我不认为这可以通过一个after_initialize钩子正确地完成。
同样,我认为“default_value_for”gem采用了正确的方法:https://github.com/FooBarWidget/default_value_for
这个问题已经回答了很长时间了,但是我经常需要默认值,并且不喜欢将它们放在数据库中。我创建了一个DefaultValues关注点:
module DefaultValues
extend ActiveSupport::Concern
class_methods do
def defaults(attr, to: nil, on: :initialize)
method_name = "set_default_#{attr}"
send "after_#{on}", method_name.to_sym
define_method(method_name) do
if send(attr)
send(attr)
else
value = to.is_a?(Proc) ? to.call : to
send("#{attr}=", value)
end
end
private method_name
end
end
end
然后像这样在我的模型中使用它:
class Widget < ApplicationRecord
include DefaultValues
defaults :category, to: 'uncategorized'
defaults :token, to: -> { SecureRandom.uuid }
end
类似的问题,但背景略有不同:
-如何为Rails activerecord的模型中的属性创建默认值?
最佳回答:这取决于你想要什么!
如果你想让每个对象都以一个值开始:使用after_initialize:init
您希望new.html表单在打开页面时具有默认值?使用https://stackoverflow.com/a/5127684/1536309
class Person < ActiveRecord::Base
has_one :address
after_initialize :init
def init
self.number ||= 0.0 #will set the default value only if it's nil
self.address ||= build_address #let's you set a default association
end
...
end
如果你想让每个对象都有一个从用户输入中计算出来的值:使用before_save:default_values
你想让用户输入X然后Y = X+'foo'?使用:
class Task < ActiveRecord::Base
before_save :default_values
def default_values
self.status ||= 'P'
end
end
Rails 5 +
你可以在你的模型中使用属性方法,例如:
class Account < ApplicationRecord
attribute :locale, :string, default: 'en'
end
您还可以将lambda传递给默认参数。例子:
attribute :uuid, :string, default: -> { SecureRandom.uuid }
第二个参数是类型,它也可以是一个自定义类型类实例,例如:
attribute :uuid, UuidType.new, default: -> { SecureRandom.uuid }
我也看到人们在迁徙中使用这种方法,但我更愿意看到
在模型代码中定义。
是否有一个规范的方法来设置字段的默认值
ActiveRecord模型?
在Rails 5之前,规范的Rails方法实际上是在迁移中设置它,只需要在db/schema中查找即可。rb每当想要看到什么默认值是由DB为任何模型设置。
与@Jeff Perrin的回答相反(这有点老),在使用Model时,迁移方法甚至会应用默认值。新的,由于一些Rails魔法。在Rails 4.1.16中验证。
最简单的东西往往是最好的。较少的知识负债和代码库中的潜在混淆点。而且它‘just works’。
class AddStatusToItem < ActiveRecord::Migration
def change
add_column :items, :scheduler_type, :string, { null: false, default: "hotseat" }
end
end
或者,在不创建新列的情况下更改列,然后执行以下操作:
class AddStatusToItem < ActiveRecord::Migration
def change
change_column_default :items, :scheduler_type, "hotseat"
end
end
或者更好:
class AddStatusToItem < ActiveRecord::Migration
def change
change_column :items, :scheduler_type, :string, default: "hotseat"
end
end
查看官方RoR指南以获得列更改方法中的选项。
null: false禁止在DB中使用null值,而且,作为一个额外的好处,它还会进行更新,以便所有先前为null的DB记录也会使用该字段的默认值设置。如果您愿意,您可以在迁移中排除这个参数,但我发现它非常方便!
Rails 5+的规范方式是,正如@Lucas Caton所说:
class Item < ActiveRecord::Base
attribute :scheduler_type, :string, default: 'hotseat'
end
这里有一个我用过的解决方案,我有点惊讶还没有加入。
它分为两部分。第一部分是在实际迁移中设置默认值,第二部分是在模型中添加验证,以确保存在为真。
add_column :teams, :new_team_signature, :string, default: 'Welcome to the Team'
您将看到这里已经设置了默认值。现在在验证中,你想要确保字符串总是有一个值,那么就这样做吧
validates :new_team_signature, presence: true
这将为您设置默认值。(对我来说,我有“Welcome to the Team”),然后它将进一步确保该对象始终存在一个值。
希望有帮助!
在开发Rails 6应用程序时,我也遇到过类似的挑战。
以下是我的解决方法:
我有一个用户表和一个角色表。Users表属于Roles表。我也有一个管理员和学生模型,从用户表继承。
然后需要在创建用户时为角色设置一个默认值,比如id = 1的管理员角色或id = 2的学生角色。
class User::Admin < User
before_save :default_values
def default_values
# set role_id to '1' except if role_id is not empty
return self.role_id = '1' unless role_id.nil?
end
end
这意味着在数据库中创建/保存admin用户之前,如果role_id不为空,则将其设置为默认值1。
return self.role_id = '1' unless role_id.nil?
等于:
return self.role_id = '1' unless self.role_id.nil?
和:
self.role_id = '1' if role_id.nil?
但是第一种方法更简洁,也更精确。
这是所有。
我希望这对你们有帮助
Rails 6.1 +
现在您可以在模型上使用属性方法,而无需设置类型。
attribute :status, default: ACTIVE
or
class Account < ApplicationRecord
attribute :locale, default: 'en'
end
注意,提供默认的to属性不能引用类的实例(lambda将在类的上下文中执行,而不是在实例中执行)。因此,如果需要根据实例或关联动态地将默认值设置为一个值,则仍然必须使用替代方法,例如after_initialize回调。如前所述,如果引用关联,建议将此限制为新记录,以避免n+1个查询。
after_initialize :do_something_that_references_instance_or_associations, if: :new_record?