PostgreSQL
 sql >> Datenbank >  >> RDS >> PostgreSQL

Rails/Postgres, „Fremdschlüssel“, die in einem Array gespeichert sind, um eine 1-viele-Assoziation zu erstellen

Sie können Rails nicht auf dieses Array aufmerksam machen und es für Assoziationen verwenden.

Wenn Sie jedoch eine schnellere Suche / Filterung von Aufgaben wünschen, die Benutzern zugewiesen sind, können Sie ein Array von Benutzer-IDs im Aufgabenobjekt speichern. Andernfalls müssten Sie einen JOIN durchführen, um alle Alice zugewiesenen Aufgaben in Ihrer Standard-Assoziationstabelle zu finden.

Die Lösung besteht also darin, die Zuordnungstabelle beizubehalten, aber auch die Benutzer-ID des zugewiesenen Benutzers in das Task-Objekt zu duplizieren und diese ID-Liste zum schnelleren Suchen/Filtern zu verwenden.

Sie müssen sich in after_create einklinken und after_destroy Lebenszyklus für die zugewiesenen Objekte und fügen Sie neue zugewiesene IDs in das Task-Datensatz-Array ein. Und dann, wenn ein Beauftragter aus einer Aufgabe entfernt wird, aktualisieren Sie das Array, um die ID zu entfernen.

Siehe Postgres-Dokumentation für alle Array-Operatoren:

Etwa so:

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