gpt4 book ai didi

ruby-on-rails - Rails 的 has_one 和 has_many 在自动数据构建方面的区别

转载 作者:太空宇宙 更新时间:2023-11-03 16:08:34 27 4
gpt4 key购买 nike

问候 StackOverflow 社区!

这几天一直在学习Rails,扩展了Rails官方指南中的基本博客示例。我想了解更多关于一对一连接的信息,因为我计划使用自定义用户数据表扩展 Devise 登录系统,并且规范化规则规定我应该为实际的身份验证数据创建一个不同的表(设计用户表)和另一个用于特定于应用程序的用户信息(我自己的用户表)。

但现在回到simple rails example .该示例基本上描述了一个可以创建博客文章、标记它们并接受评论的应用程序(评论现在并不重要)。一个帖子 has_many 标签和一个标签 belongs_to 一个帖子。非常简单。 (关于评论也可以说同样的话,但我会坚持使用这个例子的标签)

当用户希望创建新帖子时, Controller 会调用 @post = Post.new 以准备要编辑和创建的空帖子。它还需要调用 post.tags.build 以确保至少有一个属于该帖子的“baby”标签已准备好进行编辑 5.times do @post.tags.build end,例如,不会准备一个而是五个标签进行编辑。

在 Controller 的创建方法中,从 params[:post]@post.save 创建一个新帖子就足够了。它会自动保存标签,无需进行任何额外的函数调用。

这是我开始扩展的地方。我只是想添加另一个表,称为 post_data,它将与原始帖子表一对一链接。 post_data 包含其所属帖子的外键,以及其模型中的 belongs_to :post 指令。帖子 has_one :post_data。它还 accepts_nested_attibutes_for :post_data

同样,在 post Controller 的新方法中,需要初始化 post_data。@post.build_post_data 就是这样做的。表单可以很好地显示两种不同的模型 - 一种是 post[title]...等,另一种是 post_data[data_field_name_here]。

但是,在创建名称中,需要手动保存 post_data 条目:@post.create_post_data(params[:post_data]) 以使其成为进入数据库。否则它就不会被保存。

现在我的问题是 - 为什么帖子的has_many对象,标签,只需要在 Controller 的新方法中准备,然后自动保存在create中,而has_one链接还需要手动保存?

我只是想知道为什么 Rails 会这样工作。

提前感谢您的时间和耐心! :)

编辑:代码文件!

posts_controller

class PostsController < ApplicationController
# GET /posts
# GET /posts.json
def index
@posts = Post.all

respond_to do |format|
format.html # index.html.erb
format.json { render :json => @posts }
end
end

# GET /posts/1
# GET /posts/1.json
def show
@post = Post.find(params[:id])

respond_to do |format|
format.html # show.html.erb
format.json { render :json => @post }
end
end

# GET /posts/new
# GET /posts/new.json
def new
@post = Post.new
@post.build_post_data
5.times do @post.tags.build end
end

# GET /posts/1/edit
def edit
@post = Post.find(params[:id])
end

# POST /posts
# POST /posts.json
def create
# This works now - it creates all the needed resources automatically
@post = Post.new(params[:post])

respond_to do |format|
if @post.save
format.html { redirect_to @post, :notice => 'Post was successfully created.' }
else
format.html { render :action => "new" }
end
end
end

# PUT /posts/1
# PUT /posts/1.json
def update
@post = Post.find(params[:id])

respond_to do |format|
if @post.update_attributes(params[:post])
format.html { redirect_to @post, :notice => 'Post was successfully updated.' }
else
format.html { render :action => "edit" }
end
end
end

# DELETE /posts/1
# DELETE /posts/1.json
def destroy
@post = Post.find(params[:id])
@post.destroy

respond_to do |format|
format.html { redirect_to posts_url }
end
end
end

posts/_form.html.erb

<%= form_for(@post) do |post_form| %>
<% if @post.errors.any? %>
<div id="error_explanation">
<h2><%= pluralize(@post.errors.count, "error") %> prohibited this post from being saved:</h2>

<ul>
<% @post.errors.full_messages.each do |msg| %>
<li><%= msg %></li>
<% end %>
</ul>
</div>
<% end %>

<div class="field">
<%= post_form.label :name %><br />
<%= post_form.text_field :name %>
</div>


<%= post_form.fields_for(:post_data) do |pdf| %>
<div class="field">
<%= pdf.label :herp, "DERP POST DATA" %><br />
<%= pdf.text_field :herp %>
</div>
<% end %>

<div class="field">
<%= post_form.label :title %><br />
<%= post_form.text_field :title %>
</div>
<div class="field">
<%= post_form.label :content %><br />
<%= post_form.text_area :content %>
</div>

<h2>Tags</h2>
<%= render :partial => 'tags/form',
# send some custom variables to the
# rendered partial's context
:locals => { :form => post_form } %>

<div class="actions">
<%= post_form.submit %>
</div>
<% end %>

和 posts.rb 模型

class Post < ActiveRecord::Base
validates :name, :presence => true
validates :title, :presence => true, :length => { :minimum => 5 }

has_many :comments, :dependent => :destroy

attr_accessible :post_data_attributes, :name, :title, :content

#let's just assume this makes sense, k?
has_one :post_data
accepts_nested_attributes_for :post_data

# TAGS
has_many :tags
accepts_nested_attributes_for :tags, :allow_destroy => :true,
# reject if all attributes are blank
:reject_if => lambda { |attrs| attrs.all? { |k, v| v.blank? } }
end

最终编辑:

修复了所有代码,将其与我的工作代码同步! :D如果遥远 future 的任何人仍然遇到这个问题并且我的代码不起作用,请给我发送个人消息,我们会解决它! :)

最佳答案

我相信这一点:

<%= fields_for(:post_data) do |pdf|  %>

应该是这样的:

<%= post.fields_for(:post_data) do |pdf|  %>

你能试试吗?

关于ruby-on-rails - Rails 的 has_one 和 has_many 在自动数据构建方面的区别,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8902561/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com