Elixir Ecto: 多对多关系

本文是参考下面的文章做的一个实际例子
http://blog.plataformatec.com...
http://learningwithjb.com/pos...
http://stackoverflow.com/ques...
http://stackoverflow.com/ques...
http://stackoverflow.com/ques...

代码库

https://github.com/developerworks/ecto_test

定义多对多关系

创建移植脚本

mix ecto.gen.migration create_table_users -r EctoTest.Repo
mix ecto.gen.migration create_table_chat_groups -r EctoTest.Repo
mix ecto.gen.migration create_table_chat_group_users -r EctoTest.Repo

移植脚本的内容分别为:

20160429075137_create_table_users.exs

defmodule EctoTest.Repo.Migrations.CreateTableUsers do
  use Ecto.Migration

  def up do
    create table(:users) do
      add :name, :string, default: ""
      timestamps
    end

    @doc """
    Create an partial index(name != '') on `:name` column of table `users`
    """
    create index(:users, [:name], [unique: true, where: "name != ''"])
  end

  def down do
    drop table(:users)
    drop index(:users, [:name])
  end
end

20160429075138_create_table_chat_groups.exs

defmodule EctoTest.Repo.Migrations.CreateTableChatGroups do
  use Ecto.Migration

  def up do
    create table(:chat_groups) do
      add :name, :string, null: false
    end
  end

  def down do
    drop table(:chat_groups)
  end
end

20160429075146_create_table_chat_group_users.exs

defmodule EctoTest.Repo.Migrations.CreateTableChatGroupUsers do
  use Ecto.Migration

  def up do
    create table(:chat_group_users) do
      add :user_id,       references(:users)
      add :chat_group_id, references(:chat_groups)
      timestamps
    end
  end

  def down do
    drop table(:chat_group_users)
  end
end

执行移植

➜  ecto_test git:(master) ✗ mix ecto.migrate -r EctoTest.Repo

16:14:05.200 [info]  == Running EctoTest.Repo.Migrations.CreateTableChatGroupUsers.up/0 forward

16:14:05.200 [info]  create table chat_group_users

16:14:05.211 [info]  == Migrated in 0.0s

模型代码分别为

lib/ecto_test/model/user.ex

defmodule EctoTest.Model.User do
  @moduledoc """

  """
  use EctoTest.Model
  alias EctoTest.Model.ChatGroupUser

  schema "users" do
    field :name, :string, null: false
    has_many :chat_group_users, ChatGroupUser
    has_many :chats, through: [:chat_group_users, :chat]
    timestamps
  end

  @spec insert(map) :: {:ok, Schema.t} | {:error, Changeset.t}
  def insert(map) do
    Map.merge(%__MODULE__{}, map) |> Repo.insert
  end
end

lib/ecto_test/model/chat_group.ex

defmodule EctoTest.Model.ChatGroup do
  @moduledoc """

  """
  use EctoTest.Model
  alias EctoTest.Model.ChatGroupUser

  schema "chat_groups" do
    field :name, :string, null: false
    has_many :chat_group_users, ChatGroupUser
    has_many :users, through: [:chat_group_users, :user]
  end

  @spec insert(map) :: {:ok, Schema.t} | {:error, Changeset.t}
  def insert(map) do
    Map.merge(%__MODULE__{}, map) |> Repo.insert
  end
end

lib/ecto_test/model/chat_group_user.ex

defmodule ChatGroupUser do
  @moduledoc """

  """
  use EctoTest.Model
  alias EctoTest.Model.ChatGroup

  schema "chat_group_users" do
    belongs_to :chat_group, ChatGroup
    belongs_to :user, User
    timestamps
  end

  @spec insert(map) :: {:ok, Schema.t} | {:error, Changeset.t}
  def insert(map) do
    Map.merge(%__MODULE__{}, map) |> Repo.insert
  end
end

读取多对多关系

插入用户数据

iex(1)> user = %EctoTest.Model.User{name: "user1"} |> EctoTest.Repo.insert!

16:48:47.128 [debug] QUERY OK db=0.2ms
begin []

16:48:47.149 [debug] QUERY OK db=1.3ms
INSERT INTO "users" ("inserted_at","updated_at","name") VALUES ($1,$2,$3) RETURNING "id" [{{2016, 4, 29}, {8, 48, 47, 0}}, {{2016, 4, 29}, {8, 48, 47, 0}}, "user1"]

16:48:47.149 [debug] QUERY OK db=0.5ms
commit []
%EctoTest.Model.User{__meta__: #Ecto.Schema.Metadata<:loaded>,
 chat_group_users: #Ecto.Association.NotLoaded,
 chats: #Ecto.Association.NotLoaded, id: 1,
 inserted_at: #Ecto.DateTime<2016-04-29 08:48:47>, name: "user1",
 updated_at: #Ecto.DateTime<2016-04-29 08:48:47>}

插入聊天组数据

chat_group = %EctoTest.Model.ChatGroup{name: "chat_group2"} |> EctoTest.Repo.insert!
user = %EctoTest.Model.User{name: "user2"} |> EctoTest.Repo.insert!
%EctoTest.Model.ChatGroupUser{user_id: user.id, chat_group_id: chat_group.id} |> EctoTest.Repo.insert!

读取用户, 以及用户的聊天组

EctoTest.Model.User |> EctoTest.Repo.get(1) |> EctoTest.Repo.preload(:chat_groups)

或调用 EctoTest.Model.User.get(1)

EctoMix 命令

mix ecto.create        # Creates the repository storage
mix ecto.drop          # Drops the repository storage
mix ecto.dump          # Dumps database structure
mix ecto.gen.migration # Generate a new migration for the repo
mix ecto.gen.repo      # Generate a new repository
mix ecto.load          # Loads previously dumped database structure
mix ecto.migrate       # Runs the repository migrations
mix ecto.rollback      # Rolls back migrations

你可能感兴趣的:(ecto,elixir)