ruby-on-rails - HABTM 避免连接表中的欺骗

标签 ruby-on-rails ruby

我读过这个问题:has_and_belongs_to_many, avoiding dupes in the join table但我无法解决我的问题;

我有一个类TypeProduit,它有许多ExtensionFichier:

class TypeProduit < ActiveRecord::Base
    has_and_belongs_to_many :extension_fichiers
end

和:

class ExtensionFichier < ActiveRecord::Base
  has_and_belongs_to_many :type_produits
end

它们是相关的

class LinkExtensionFichiers < ActiveRecord::Migration
  def change
     create_join_table :extension_fichiers, :type_produits
  end
end

编辑:

这是[正确的] Controller :

class TypeProduitsController < ApplicationController
before_action :set_type_produit, only: [:show, :edit, :update, :destroy]

# GET /type_produits
# GET /type_produits.json
def index
  @type_produits = TypeProduit.all
end

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

# GET /type_produits/new
def new
  @type_produit = TypeProduit.new
end

# GET /type_produits/1/edit
def edit
end

# POST /type_produits
# POST /type_produits.json
def create
  @type_produit = TypeProduit.new(type_produit_params)

  puts "------extension_fichier_params------"
  puts extension_fichier_params

  #@type_produit.extension_fichier_ids = extension_fichier_params

  @type_produit.extension_fichiers << ExtensionFichier.find_by(:id => extension_fichier_params)

  respond_to do |format|
    if @type_produit.save
      format.html { redirect_to @type_produit, notice: 'Type produit was successfully created.' }
      format.json { render :show, status: :created, location: @type_produit }
    else
      format.html { render :new }
      format.json { render json: @type_produit.errors, status: :unprocessable_entity }
    end
  end
end

# PATCH/PUT /type_produits/1
# PATCH/PUT /type_produits/1.json
def update
  respond_to do |format|
    if @type_produit.update(type_produit_params)
      format.html { redirect_to @type_produit, notice: 'Type produit was successfully updated.' }
      format.json { render :show, status: :ok, location: @type_produit }
    else
      format.html { render :edit }
      format.json { render json: @type_produit.errors, status: :unprocessable_entity }
    end
  end
end

# DELETE /type_produits/1
# DELETE /type_produits/1.json
def destroy
  @type_produit.destroy
  respond_to do |format|
    format.html { redirect_to type_produits_url, notice: 'Type produit was successfully destroyed.' }
    format.json { head :no_content }
  end
end

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

  def extension_fichier_params
    @type_produit.extension_fichier_ids
  end

  # Never trust parameters from the scary internet, only allow the white list through.
  def type_produit_params
    params.require(:type_produit).permit(:nom, :extension_fichier_ids => [])
  end
end

这是 Rails 服务器输出:

Started POST "/type_produits" for 127.0.0.1 at 2015-01-27 17:55:15 +0100
Processing by TypeProduitsController#create as HTML
  Parameters: {"utf8"=>"✓", "authenticity_token"=>"4LBhwTvGjXOq+3qzO+loGUV/oBCYjW66aVDeE1Zj/AM=", "type_produit"=>{"nom"=>"qsdfqsdf", "extension_fichier_ids"=>["1", "2", ""]}, "commit"=>"Create Type produit"}
  ExtensionFichier Load (0.4ms)  SELECT "extension_fichiers".* FROM "extension_fichiers"  WHERE "extension_fichiers"."id" IN (1, 2)
------extension_fichier_params------
1
2
  ExtensionFichier Load (0.2ms)  SELECT  "extension_fichiers".* FROM "extension_fichiers"  WHERE "extension_fichiers"."id" IN (1, 2) LIMIT 1
   (0.1ms)  begin transaction
  TypeProduit Exists (0.1ms)  SELECT  1 AS one FROM "type_produits"  WHERE "type_produits"."nom" = 'qsdfqsdf' LIMIT 1
  SQL (0.4ms)  INSERT INTO "type_produits" ("created_at", "nom", "updated_at") VALUES (?, ?, ?)  [["created_at", "2015-01-27 16:55:15.923132"], ["nom", "qsdfqsdf"], ["updated_at", "2015-01-27 16:55:15.923132"]]
  SQL (0.1ms)  INSERT INTO "extension_fichiers_type_produits" ("extension_fichier_id", "type_produit_id") VALUES (?, ?)  [["extension_fichier_id", 1], ["type_produit_id", 8]]
  SQL (0.1ms)  INSERT INTO "extension_fichiers_type_produits" ("extension_fichier_id", "type_produit_id") VALUES (?, ?)  [["extension_fichier_id", 2], ["type_produit_id", 8]]
  SQL (0.0ms)  INSERT INTO "extension_fichiers_type_produits" ("extension_fichier_id", "type_produit_id") VALUES (?, ?)  [["extension_fichier_id", 1], ["type_produit_id", 8]]
   (0.5ms)  commit transaction

为什么 Rails 只包含 2 个对象,却要插入 3 个对象

最佳答案

您的 Controller 代码中有一个非常明显的错误。让我们一步步浏览您的create行动。

您创建一个新的TypeProduit来自type_produit_params其中包括:nom :extension_fichier_ids 。然后输出 extension_fichier_params 的值它本身是 @type_produit.extension_fichier_ids你得到 "1\n2\n"所以你的@type_produit已经拥有这两个 extension_fichier已附加到它。

然后附加 ExtensionFichier.find_by(:id => extension_fichier_params)@type_produit.extension_fichier_ids 。现在是find_by基本上是一个 where().first所以它只得到第一个,即。 id=1。所以现在在你的 @type_produit.extension_fichiers array 你又得到了 id 1、2 和 1。 IE。三个对象。

然后保存TypeProduit所有这三个关联都与三个 INSERT 一起保存。 s。

是的,对此的整体解决方案(除了修复不必要的 << 代码或添加 1 和 2 的模型或 View 之外)是添加 -> { uniq }根据 https://stackoverflow.com/a/20226630/152786 涵盖您的 Habtm 协会的范围

关于ruby-on-rails - HABTM 避免连接表中的欺骗,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28176158/

相关文章:

ruby-on-rails - 按属性的特定值或值数组排序

ruby-on-rails - Rails i18n 前端、后端的不同语言环境

ruby - 将字符串拆分为文本和整数

ruby-on-rails - 我怎样才能在 ActiveAdmin/Arbre 中有一个有多个 child 的 div

python - Ruby 替代 Scrapy?

ruby-on-rails - 未定义的方法!在 rails 上使用MiniMagick的Carrierwave 3.1.3

ruby-on-rails - 致命 : Peer authentication failed for user "rails"

ruby - 使用 # 代替 .在 API 文档中

ruby-on-rails - 使用 POST/sign_in 登录。不支持 GET。 Omniauth-facebook

javascript - 使用提交按钮将 plan_id 传递到参数中