Ruby on rails HABTM避免联接表中的重复
我读过这个问题:但我无法解决我的问题 我有一个类Ruby on rails HABTM避免联接表中的重复,ruby-on-rails,ruby,Ruby On Rails,Ruby,我读过这个问题:但我无法解决我的问题 我有一个类TypeProduit,它有许多extensionChier: class TypeProduit < ActiveRecord::Base has_and_belongs_to_many :extension_fichiers end 为什么rails只包含2个对象却要进行3个insert您是否尝试了所建议的解决方案(可能还会按照中的建议添加索引?控制器代码中有一个非常明显的错误。让我们一步一步地完成您的create操作 您可以从
TypeProduit
,它有许多extensionChier
:
class TypeProduit < ActiveRecord::Base
has_and_belongs_to_many :extension_fichiers
end
为什么rails只包含2个对象却要进行3个insert您是否尝试了所建议的解决方案(可能还会按照中的建议添加索引?控制器代码中有一个非常明显的错误。让我们一步一步地完成您的
create
操作
您可以从type_produit_params
创建一个新的TypeProduit
,其中包括:nom
和:extension\u fichier\u id
。然后输出extension\u fichier\u params
的值,该值本身就是@type\u produit.extension\u fichier\u id
,然后得到“1\n2\n”
,因此@type\u produit
已经附加了这两个扩展\u fichier
然后将extensionChier.find\u by(:id=>extension\u fichier\u params)
附加到@type\u produit.extension\u fichier\u id
。现在,find_by
基本上是一个where()。first
因此它只得到第一个,即id=1。因此,现在在@type\u produit.extension\u fichiers
数组中,您又有了ID 1、2和1。三个物体
然后保存TypeProduit
,所有这三个关联都与三个INSERT
s一起保存
是的,这个问题的整体解决方案(除了修复不必要的
之外,您正在显示用于创建TypeProduit
对象的模型和日志,但是您的控制器是用于创建Produit
对象的代码。要正确诊断此问题,我们需要查看TypeProduitsController
@smathy您是对的,愚蠢的复制/粘贴。现在您的code看起来不错,您应该检查您的extension fichiers表中没有id 1的重复记录:extensionChier.find by(:id=>[1,2])
这似乎不太可能。@ThomasGuillory数据库中没有重复:/第二个链接是我在问题中提到的链接,uniq:true
不起作用,但谢谢!
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
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