Warning: file_get_contents(/data/phpspider/zhask/data//catemap/5/ruby/25.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Ruby on rails HABTM避免联接表中的重复_Ruby On Rails_Ruby - Fatal编程技术网

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