《【译】rails的嵌套属性(Nested Attributes)使用.pdf》由会员分享,可在线阅读,更多相关《【译】rails的嵌套属性(Nested Attributes)使用.pdf(7页珍藏版)》请在得力文库 - 分享文档赚钱的网站上搜索。
1、【译】rails 的嵌套属性(Nested Attributes)使用Active Record Nested Attributes通过嵌套属性(nested attribute),你可以通过 parent 来保存与其相关联的属性。默认情况下,嵌套属性是关闭的,你可以开启 accepts_nested_attributes_for 这个类方法,就在该 model 上生成一个属性 writer。属性 writer 是以该关联命名。例如,为你的model 增加两个新方法:author_attributes=(attributes)和pages_attributes=(attributes).Rub
2、y 代码class Book<ActiveRecord:Basehas_one:authorhas_many:pagesaccepts_nested_attributes_for:author,:pagesend使用了 accepts_nested_attributes_for 的每一个关联都自动开启:autosave一对一关联一个 Member 有一个 AvatarRuby 代码class Member<ActiveRecord:Basehas_one:avataraccepts_nested_attributes_for:avatarend开启一对一关联的嵌套属性可以通过这样方
3、法一次性创建Member:Ruby 代码params=:member=>:name=>Jack,:avatar_attributes=>:icon=>smiling member=Member.create(params:member)member.avatar.id#=>2member.avatar.icon#=>smiling也可以这样 update avatarparams=:member=>:avatar_attributes=>:id=>2,:icon=>sad member.update_attributesparams:
4、membermember.avatar.icon#=>sad默认情况下你只能设置或更新关联的 model。如果你想通过属性 hash 来删除关联 model,你需要使用:allow_destroy 选项Ruby 代码class Member<ActiveRecord:Basehas_one:avataraccepts_nested_attributes_for:avatar,:allow_destroy=>trueend此时,如果向属性 hash 里增加一个_destroy 的 key,并且value 是 true,则该关联 model 将被删除。Ruby 代码member.
5、avatar_attributes=:id=>2,:_destroy=>1 member.avatar.marked_for_destruction?#=>truemember.savemember.reload.avatar#=>nil注意这里,只有当 parent 被保存后,关联的 model 才真正被删除。一对多关联一个 member 有一些 postRuby 代码class Member<ActiveRecord:Basehas_many:postsaccepts_nested_attributes_for:postsend你可通过属性 hash 来增加或
6、更新关联 post model每一个不含有 id 键的新记录会被实例化,除非该 hash 也包含了一个:_destroy=>trueRuby 代码params=:member=>:name=>joe,:posts_attributes=>:title=>Kari,the awesome Rubydocumentation browser!,:title=>Theegalitarian assumption of the modern citizen,:title=>,:_destroy=>1#该记录会被忽略member=Member.create
7、(paramsmember)member.posts.length#=>2member.posts.first.title#=>Kari,theawesome Ruby documentation browser!member.posts.second.title#=>The egalitarianassumption of the modern citizen通过:reject_if proc 设置忽略的不满足条件的记录。例如:Ruby 代码class Member<ActiveRecord:Basehas_many:postsaccepts_nested_attri
8、butes_for:posts,:reject_if=>proc|attributes|attributestitle.blank?endparams=:member=>:name=>joe,:posts_attributes=>:title=>Kari,the awesome Rubydocumentation browser!,:title=>Theegalitarian assumption of the modern citizen,:title=>#这个记录会被忽略member=Member.create(paramsmember)membe
9、r.posts.length#=>2member.posts.first.title#=>Kari,theawesome Ruby documentation browser!member.posts.second.title#=>The egalitarianassumption of the modern citizen:reject_if 也可一接受一个 symbol 来代表一个可用方法:如果 hash 中含有一个 id 和已有的关联记录相匹配,则被匹配到的记录会被修改:Ruby 代码member.attributes=:name=>Joe,:posts_attr
10、ibutes=>:id=>1,:title=>UPDATED An,as of yet,undisclosed awesomeRuby documentation browser!,:id=>2,:title=>UPDATED other post member.posts.first.title#=>UPDATED An,as of yet,undisclosed awesome Ruby documentation browser!member.posts.second.title#=>UPDATED other post默认关联记录是被保护的(不
11、被删除)。如果想通过属性 hash来删除任何关联记录,你需要打开:allow_destroy 选项,这样使用_destroy 键来删除记录就行:Ruby 代码class Member<ActiveRecord:Basehas_many:postsaccepts_nested_attributes_for:posts,:allow_destroy=>trueendparams=:member=>:posts_attributes=>:id=>2,:_destroy=>1 member.attributes=paramsmembermember.posts.de
12、tect|p|p.id=2.marked_for_destruction?#=>true标记将被删除member.posts.length#=>2未保存,所以还没删除member.savemember.reload.posts.length#=>1保存所有对model修改的行为,包括标记将要销毁,会随着parent被保存原子性的被自动保存或者删除。这些开始与 parent的 save 方法,发生在其内部的事务里。详见 Active RecordAutosave Association使用 attr_accessible使用 attr_accessible 如果不小心,可能会干
13、扰到嵌套属性的使用。例如,上述 member model 如下使用 attr_accessible:Ruby 代码attr_accessible:name你需要这样的修改:Ruby 代码attr_accessible:name,:posts_attributes验证 parent model 的存在如果你想验证一个 child 记录是否和一个 parent 记录关联,你可以使用validates_presence_of 和 inverse_of:Ruby 代码<span style=font-size:xx-small;>classMember<ActiveRecord:Basehas_many:posts,:inverse_of=>:memberaccepts_nested_attributes_for:postsendclass Post<ActiveRecord:Basebelongs_to:member,:inverse_of=>:postsvalidates_presence_of:memberend</span>具体应用见http:/