活动记录关联未保存在数据库中

Posted

技术标签:

【中文标题】活动记录关联未保存在数据库中【英文标题】:Active Record Association not saving in database 【发布时间】:2015-04-09 21:48:58 【问题描述】:

我有两个模型:

class Tool < ActiveRecord::Base
    belongs_to :user
end

class User < ActiveRecord::Base
  has_many :tools, dependent: :destroy
end

我创建了一个迁移来提供模型的外键:

class AddUserIdToTool < ActiveRecord::Migration
  def change
    add_column :tools, :user_id, :integer
  end
end

我有以下表格:

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

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

  <div class="field">
    <%= f.label :name %><br>
    <%= f.text_field :name %>
  </div>
  <div class="field">
    <%= f.label :description %><br>
    <%= f.text_area :description %>
  </div>
  <div class="field">
    <%= f.file_field :tool_image %>
  </div>
  <div class="actions">
    <%= f.submit %>
  </div>
<% end %>

每当我提交表单时,外键都不会更新(user_id)。我在控制台中看到以下内容:

irb(main):002:0> Tool.last
  Tool Load (0.0ms)  SELECT  "tools".* FROM "tools"  ORDER BY "tools"."id" DESC LIMIT 1
=> #<Tool id: 6, name: "wrench2", description: "another wrench", created_at: "2015-04-09 21:38:47", updated_at: "2015-04
-09 21:38:47", tool_image_file_name: "wrench.jpg", tool_image_content_type: "image/jpeg", tool_image_file_size: 3424, to
ol_image_updated_at: "2015-04-09 21:38:43", user_id: nil>
irb(main):003:0>

如您所见,user_idnil

这是我的控制器:

class ToolsController < ApplicationController
  before_action :authenticate_user!, only: [:new, :destroy, :edit, :update], notice: 'you must sign in first!'
  before_action :set_tool, only: [:show, :edit, :update, :destroy]

  # GET /tools
  # GET /tools.json
  def index
    @tools = Tool.all
  end

  # GET /tools/1
  # GET /tools/1.json
  def show
  end

  # GET /tools/new
  def new
    @tool = Tool.new
  end

  # GET /tools/1/edit
  def edit
  end

  # POST /tools
  # POST /tools.json
  def create
    @tool = Tool.new(tool_params)

    respond_to do |format|
      if @tool.save
        format.html  redirect_to @tool, notice: 'Tool was successfully created.' 
        format.json  render :show, status: :created, location: @tool 
      else
        format.html  render :new 
        format.json  render json: @tool.errors, status: :unprocessable_entity 
      end
    end
  end

  # PATCH/PUT /tools/1
  # PATCH/PUT /tools/1.json
  def update
    respond_to do |format|
      if @tool.update(tool_params)
        format.html  redirect_to @tool, notice: 'Tool was successfully updated.' 
        format.json  render :show, status: :ok, location: @tool 
      else
        format.html  render :edit 
        format.json  render json: @tool.errors, status: :unprocessable_entity 
      end
    end
  end

  # DELETE /tools/1
  # DELETE /tools/1.json
  def destroy
    @tool.destroy
    respond_to do |format|
      format.html  redirect_to tools_url, notice: 'Tool was successfully destroyed.' 
      format.json  head :no_content 
    end
  end

  private
    # Use callbacks to share common setup or constraints between actions.
    def set_tool
      @tool = Tool.find(params[:id])
    end

    # Never trust parameters from the scary internet, only allow the white list through.
    def tool_params
      params.require(:tool).permit(:name, :description, :tool_image, :user_id)
    end

end

如何使用登录用户的 ID 更新 user_id?我是否必须在我的表单中包含一个隐藏字段或类似的东西?

我还必须在我的控制器的强参数中允许 user_id 变量吗?这个我试过了,还是没解决问题...

这是如何工作的?

【问题讨论】:

您还没有包含最重要的部分,即控制器中实际处理表单提交数据的操作。 @meagar 我已经用控制器代码更新了我的问题。嗯,也许我需要在我的控制器中添加一些逻辑。类似于:@user.tools.build(params[:user][:tool]) 这种东西在 Rails 中的约定是什么? 【参考方案1】:

目前您正在自己构建一个工具。要将其与用户模型相关联,您需要按以下方式创建工具模型:

@user.tools.build

它将使用所属模型的正确 user_id 创建一个新的 ActiveRecord 工具模型。但是要获得父母,您需要从您的参数中获取其 id。在你的行动中,它可以以这样的方式完成:

@user = User.find(params[:user_id])

注意:为了让您的路线可以相应嵌套(在此处阅读更多信息:http://guides.rubyonrails.org/routing.html)

因此,总而言之,您的新操作应如下所示:

def new
  @user = User.find(params[:user_id])
  @tool = @user.tools.build
end

在你的 routes.rb 中

resources :users do
  resources :tools
end

【讨论】:

如果我不想嵌套路由怎么办?这会改变我应该做的事情吗?我在任何地方都找不到任何信息。 @coderwannabe2,基本上你只需要知道用户(或其ID)。在嵌套路由中,很容易从参数中获取其 id。但您也可以手动将其输入到表单中。那么您在新操作@user = User.find(params[:user_id]) 中将不需要这行代码。但是,您需要为 user_id 创建一个字段,并输入用户的 id。或者,您可以使用选择字段从现有用户列表中进行选择。 不创建字段可以吗?这似乎不是一个优雅的解决方案......我相信一定有一种方法看起来像这样一个常见的用例。 @coderwannabe2 想想你的工具模型如何知道它应该属于哪个用户?可以有 1 或 100 或 1 000 000 个用户。他们谁应该有那个工具? 你能给我看一个代码例子吗?我想我可能误解了你在说什么。我不想拥有嵌套资源,但不知道如何将工具附加到没有嵌套或没有 form_field 的用户。

以上是关于活动记录关联未保存在数据库中的主要内容,如果未能解决你的问题,请参考以下文章

oracle添加记录的时候提示违反完整性约束,未找到父项关键字怎么解决

CakePHP - 如何在保存时将关联记录添加到模型中?

在 CakePHP 中保存新的关联记录

如何在整个 Rails 活动记录关联链中取消特定模型的默认范围?

使用一个整数字段保存多个状态位数据

使用 python 记录未保存的 Excel 数据或使用 vba 自动保存