我想通过在ActiveRecord中定义属性来为其创建默认值。默认情况下,每次创建记录时,我都想拥有属性:status
的默认值。我试着这样做:
class Task < ActiveRecord::Base
def status=(status)
status = 'P'
write_attribute(:status, status)
end
end
但是在创建时我仍然从数据库中检索到这个错误:
ActiveRecord::StatementInvalid: Mysql::Error: Column 'status' cannot be null
因此我假设该值未应用于该属性。
在Rails中执行此操作的优雅方法是什么?
非常感谢。
答案 0 :(得分:280)
您可以在迁移
中为列设置默认选项....
add_column :status, :string, :default => "P"
....
OR
您可以使用回调before_save
class Task < ActiveRecord::Base
before_save :default_values
def default_values
self.status ||= 'P' # note self.status = 'P' if self.status.nil? might be safer (per @frontendbeauty)
end
end
答案 1 :(得分:188)
因为我刚刚遇到这个问题,并且Rails 3.0的选项有点不同,我将提供另一个问题的答案。
在Rails 3.0中,您希望执行以下操作:
class MyModel < ActiveRecord::Base
after_initialize :default_values
private
def default_values
self.name ||= "default value"
end
end
答案 2 :(得分:81)
当我需要默认值时,它通常用于渲染新操作视图之前的新记录。以下方法将仅为新记录设置默认值,以便在呈现表单时可用。 before_save
和before_create
为时已晚,如果您希望默认值显示在输入字段,则无效。
after_initialize do
if self.new_record?
# values will be available for new record forms.
self.status = 'P'
self.featured = true
end
end
答案 3 :(得分:77)
您可以在不编写任何代码的情况下执行此操作:)您只需在数据库中设置列的默认值即可。您可以在迁移中执行此操作。例如:
create_table :projects do |t|
t.string :status, :null => false, :default => 'P'
...
t.timestamps
end
希望有所帮助。
答案 4 :(得分:23)
解决方案取决于一些事情。
默认值是否依赖于创建时可用的其他信息? 你可以用最小的后果擦除数据库吗?
如果您回答第一个问题,那么您想使用Jim的解决方案
如果您回答第二个问题,那么您想使用Daniel的解决方案
如果对这两个问题的回答都是“否”,那么最好添加和运行新的迁移。
class AddDefaultMigration < ActiveRecord::Migration
def self.up
change_column :tasks, :status, :string, :default => default_value, :null => false
end
end
:string可以替换为ActiveRecord :: Migration识别的任何类型。
CPU很便宜,所以在Jim的解决方案中重新定义Task不会导致很多问题。特别是在生产环境中。这种迁移是正确的方式,因为它加载它并且调用次数要少得多。
答案 5 :(得分:12)
我会考虑使用找到的here attr_defaults。你最疯狂的梦想会实现。
答案 6 :(得分:10)
使用presence可以做
class Task < ActiveRecord::Base
before_save :default_values
def default_values
self.status = status.presence || 'P'
end
end
答案 7 :(得分:4)
对于列类型,Rails支持开箱即用 - 就像这个问题中的字符串一样 - 最好的方法是在Daniel Kristensen指出的数据库中设置列默认值。 Rails将对DB进行内省并相应地初始化对象。另外,这使得您的数据库安全,因为有人在您的Rails应用程序之外添加一行而忘记初始化该列。
对于列类型,Rails不支持开箱即用 - 例如ENUM列 - Rails将无法内省列默认值。对于这些情况,你不想要使用after_initialize(每次从DB加载对象时调用它,以及每次使用.new创建对象时调用),before_create(因为它发生在验证)或before_save(因为它也会在更新时发生,通常不是你想要的)。
相反,您希望在before_validation中设置属性:create,如下所示:
before_validation :set_status_because_rails_cannot, on: :create
def set_status_because_rails_cannot
self.status ||= 'P'
end
答案 8 :(得分:4)
正如我所看到的,当需要默认值时,需要解决两个问题。
这是我的解决方案:
# the reader providers a default if nil
# but this wont work when saved
def status
read_attribute(:status) || "P"
end
# so, define a before_validation callback
before_validation :set_defaults
protected
def set_defaults
# if a non-default status has been assigned, it will remain
# if no value has been assigned, the reader will return the default and assign it
# this keeps the default logic DRY
status = status
end
我很想知道为什么人们会想到这种方法。
答案 9 :(得分:0)
我现在找到了一种更好的方法:
def status=(value)
self[:status] = 'P'
end
在Ruby中,允许方法调用没有括号,因此我应该将局部变量命名为其他内容,否则Ruby会将其识别为方法调用。