我正在尝试添加一个前置过滤器来操作用户输入:
class FormField < ActiveRecord::Base
attr_accessible :field_type, :name, :field_options
serialize :field_options, Array
before_validation :update_field_options
def update_field_options
begin
self.field_options = self.field_options.split(/\,/).map(&:strip)
rescue ActiveRecord::SerializationTypeMismatch
errors.add_to_base("Die Optionen bitte mit Kommata trennen)")
false
end
end
end
当我尝试使用“1,2,3”创建记录时,它不起作用(ActiveRecord :: SerializationTypeMismatch)。
迁移是:
t.string :field_options, :null => true
不确定我在这里做错了什么。如果我将之前的过滤器更改为:
self.field_options = [1,2,3]
它有效,似乎我无法访问self.field_options,但我可以设置它们.....
logger.warn { "field_options #{ self.field_options }" }
不会输出任何内容,甚至不会输出“field_options”,self会给我:
类似于:
Ruby on Rails 3 - Serialize Array Mismatched Class Bafflement
LOG:
Started POST "/global/form_fields?format=" for 127.0.0.1 at Sun Mar 20 17:50:00 +0100 2011
SQL (3.8ms) describe `shift_categories_users`
SQL (3.9ms) describe `roles_users`
Processing by Global::FormFieldsController#create as
Parameters: {"form_field"=>{"name"=>"rails verion", "field_options"=>"1,2,3", "field_type"=>"select"}, "commit"=>"Create Form field", "authenticity_token"=>"nTmPr1H3Ilp6eRqLq/9Gd0JZx7wAw0lHqGlMBEq74HU=", "utf8"=>"✓"}
User Load (5.0ms) SELECT `users`.* FROM `users` WHERE `users`.`id` = 261 LIMIT 1
SQL (0.9ms) BEGIN
SQL (0.4ms) ROLLBACK
Account Load (1.6ms) SELECT `accounts`.* FROM `accounts` WHERE `accounts`.`subdomain` = 'expertcloud' LIMIT 1
ConfigOption Load (1.8ms) SELECT `config_options`.* FROM `config_options` WHERE `config_options`.`name` = 'headline' AND (`config_options`.account_id = 82) LIMIT 1
DEPRIATION WARNING - make sure you use it in the correct context. user.admin? use user.role?(:admin) insted
DEPRIATION WARNING - make sure you use it in the correct context. user.admin? use user.role?(:admin) insted
SQL (3.1ms) describe `roles_users`
Role Load (2.0ms) SELECT `roles`.* FROM `roles`
Role Load (2.0ms) SELECT * FROM `roles` INNER JOIN `roles_users` ON `roles`.id = `roles_users`.role_id WHERE (`roles_users`.user_id = 261 ) LIMIT 1
Rendered shared/_options.html.haml (73.3ms)
CACHE (0.0ms) SELECT `config_options`.* FROM `config_options` WHERE `config_options`.`name` = 'headline' AND (`config_options`.account_id = 82) LIMIT 1
Role Load (1.2ms) SELECT `roles`.* FROM `roles` INNER JOIN `roles_users` ON `roles`.id = `roles_users`.role_id WHERE `roles`.`name` = 'admin' AND (`roles_users`.user_id = 261 ) LIMIT 1
SQL (2.9ms) describe `shift_categories_users`
Plan Load (1.8ms) SELECT `plans`.* FROM `plans` WHERE `plans`.`id` = 4 LIMIT 1
Rendered shared/_navigation.html.haml (179.3ms)
Rendered shared/_header.html.haml (269.9ms)
Rendered shared/_footer.html.haml (1.1ms)
Rendered global/form_fields/new.html.haml within layouts/application (1141.4ms)
Completed 200 OK in 1354ms (Views: 1231.8ms | ActiveRecord: 33.0ms)
修改 在控制台中:
FormField.create!(:field_options => "1,2,3")
如果我尝试在模型中调试它:
puts "SPLIT #{ self.field_options }"
puts "SPLIT #{ self.inspect }"
在阅读 self.field_options (即“1,2,3”)时出现错误
答案 0 :(得分:0)
嗯..只是对此嗤之以鼻,也许你的field_options应该是二进制而不是字符串
t.binary :field_options, :null => true
答案 1 :(得分:0)
您声明field_options必须从数组继承。
serialize :field_options, Array
当你为field_options分配一个数组时,它会为你序列化数据(在yaml中,但这对你来说应该是透明的。)
你不能:
self.field_options = "any string"
因为字符串不是数组。如果必须预处理用户输入字符串,则需要
attr_accessible :field_options_str # Rails mass-assign
attr_accessor :field_options_str # Ruby
def update_field_options
self.field_options = self.field_options_str.split(/\,/).map(&:strip)
# I had to…
# field_options = field_options_str.split(',').map &:strip
end
你也可以,
serialize :field_options # remove Array
然后,rails会将您的字符串序列化为yaml字符串,self.field_options会将其反序列化为ruby字符串,允许您拆分并映射它,并存储生成的数组。