HABTM避免连接表中的欺骗

时间:2015-01-27 17:17:29

标签: 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

编辑:

这是[正确的]控制器:

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只有3个插入,而它只有2个对象

2 个答案:

答案 0 :(得分:2)

您的控制器代码中存在一个非常明显的错误。让我们一步一步地完成您的create操作。

您从TypeProduit创建了一个新的type_produit_params,其中包含:nom :extension_fichier_ids。然后,您输出extension_fichier_params的值@type_produit.extension_fichier_ids,然后获得"1\n2\n",以便@type_produit已经附加了extension_fichierExtensionFichier.find_by(:id => extension_fichier_params)。< / p>

然后,您将@type_produit.extension_fichier_ids追加到find_by。现在where().first基本上是@type_produit.extension_fichiers所以它只得到第一个,即。 ID = 1。所以现在在你的TypeProduit数组中,你再次有id 1,2和1。 IE浏览器。三个对象。

然后保存INSERT,并将所有这三个关联保存为三个<<

是的,对此的整体解决方案(除了修复您不必要的-> { uniq }代码或您添加1和2的模型或视图)是添加{{1}}范围根据{{​​3}}

到您的habtm协会

答案 1 :(得分:0)

您是否尝试过using-uniq-in-a-has-and-belongs-to-many-relationship-in-rails-4建议的解决方案(也可能按has-and-belongs-to-many-avoiding-dupes-in-the-join-table中的建议添加索引?)