11

postgres配列を使用して、レール(4)で1対多/has_many関連を作成できますか? 外部キー型の配列ができないことは承知しています。

例: タスクには複数の担当者がいます。伝統的に、関連テーブルを使用してこれを解決します: タスク -> 担当者 -> ユーザー。配列を使用すると、複数の「外部キー」を格納できるため、これは必要ありません。

次に、次のクエリを使用して、自分に割り当てられたすべてのタスクを取得できます。

select * from tasks where ? IN tasks.assignees
4

1 に答える 1

6

Rails にこの配列を認識させて関連付けに使用することはできません。

しかし、ユーザーに割り当てられたタスクをより迅速に検索/フィルタリングしたい場合は、Task オブジェクトにユーザー ID の配列を保持できます。そうしないと、標準の関連付けテーブルで Alice に割り当てられたすべてのタスクを見つけるために JOIN を実行する必要があります。

したがって、解決策は、関連付けテーブルを保持するだけでなく、担当者のユーザー ID を Task オブジェクトに複製し、その ID リストを使用して検索/フィルタリングを高速化することです。

担当者オブジェクトのafter_createとライフサイクルにフックし、新しい担当者 ID をタスク レコード配列に挿入する必要があります。after_destroyそして、担当者がタスクから削除されたら、配列を更新して ID を削除します。

すべての配列演算子については、Postgres のドキュメントを参照してください。

このようなもの:

class Task < ActiveRecord::Base
    has_many :assignees, :dependent => :destroy
end

class Asignee < ActiveRecord::Base

    belongs_to :task
    after_create :insert_task_assignee
    after_destroy :remove_task_assignee

    # assumes that there is a column called assignee_id
    # that contains the User ID of the assigned person

    private

    def insert_task_assignee
        # TODO: check for duplicates here - before we naively push it on?
        task.assignee_list = task.assignee_list.push(assignee_id)
        task.assignee_list.save
    end

    def remove_task_assignee
        id_list = task.assignee_list
        id_list.reject! { |candidate_id| candidate_id == assignee_id }
        task.assignee_list = id_list
        task.assignee_list.save
    end

end

# find all tasks that have been assigned Alice & Bob
# this will require the `postgres_ext` gem for Ruby / Postgres array searching
tasks = Task.where.contains(:assignee_list => [alice.id, bob.id]).all
于 2013-06-09T17:35:32.787 に答える