SHOEISHA iD

※旧SEメンバーシップ会員の方は、同じ登録情報(メールアドレス&パスワード)でログインいただけます

CodeZine編集部では、現場で活躍するデベロッパーをスターにするためのカンファレンス「Developers Summit」や、エンジニアの生きざまをブーストするためのイベント「Developers Boost」など、さまざまなカンファレンスを企画・運営しています。

IDDD本から理解するドメイン駆動設計

実践DDD本 第12章「リポジトリ」~集約の永続化管理を担当~

IDDD本から理解するドメイン駆動設計 第12回

  • X ポスト
  • このエントリーをはてなブックマークに追加

リポジトリを利用する「アプリケーションサービス」

 それでは、リポジトリを利用するコード例(JavaC#)から見ていきましょう。以下では、ユーザー管理のアプリケーションサービス(14章)にて、「ユーザー」の集約をデータベースに登録しています。アプリケーションサービスはUIや外部サービスのユースケースから呼び出されます。

[リスト]リポジトリを呼び出すアプリケーションサービス(Java)
// ■ユーザーIDに関するアプリケーションサービス
public class IdentityApplicationService {

    // ▼リポジトリ(SpringのDIにて設定)
    @Autowired
    private UserRepository userRepository;

    // ▼ユーザーを登録する(トランザクション宣言をアノテーション指定)
    @Transactional
    public User registerUser(RegisterUserCommand aCommand) {

        Tenant tenant = this.existingTenant(aCommand.getTenantId());

        // ユーザーの集約を生成(テナント集約のファクトリから生成)
        User user =
            tenant.registerUser(
                    aCommand.getInvitationIdentifier(),
                    aCommand.getUsername(),
                   ~中略~
                   )

        // ユーザーをリポジトリに登録
        this.userRepository().add(user);
        return user;
    }

 上記のコードにて、ファクトリが集約を生成しリポジトリへ登録する流れが理解できたと思います。

リポジトリのモジュール構成

 続けて、リポジトリのモジュール(パッケージ/名前空間)構成を整理してみましょう。

リポジトリに関するレイヤ/モジュール構成
リポジトリに関するレイヤ/モジュール構成

 上図の通り、リポジトリのインターフェイスは、集約と同じドメイン層のパッケージに配置します。そしてリポジトリの実装クラスはインフラ層に配置します。4章「アーキテクチャ」で紹介したDIP原則に従うことで、ドメイン層への影響なく、リポジトリの交換が可能になります。

データベースのトランザクション管理

 なお、リポジトリではデータベースのトランザクション管理は行いません。そのため、アプリケーションサービス側にてトランザクションを管理する必要があります。トランザクションは、明示的にメソッドで呼び出す場合もあれば、宣言的に管理する場合もあります。上記コードでは「@Transactional」アノテーションにて宣言的にトランザクションを制御しています。

リポジトリの設計 ~「コレクション指向」と「永続指向」~

 続いて、リポジトリの設計について見ていきましょう。IDDDでは2つの設計パターンを紹介しています。ひとつはRDBやインメモリで格納する「コレクション指向リポジトリ」で、もうひとつはNoSQLに格納する「永続指向リポジトリ」です。

IDDDにおける「リポジトリ」の実装方式(2つ)
IDDDにおける「リポジトリ」の実装方式(2つ)

 「コレクション指向のリポジトリ」では集約の登録時に「追加メソッド(Add)」を用います。リポジトリは、登録された集約の情報をRDBに書き込みます。IDDDではO/Rマッパー(Hibernate等)の仕組みでオブジェクトの変更を検知し、複数のテーブルに反映します。

 これに対して「永続指向のリポジトリ」では集約の作成/変更時に「保存メソッド(Save)」を呼び出します。リポジトリでは、保存メソッドが呼ばれたタイミングで、集約のキーを元に、NoSQLに保存します。集約の階層構造を1つのデータ(ドキュメント)として保存します。そのため、差分的な変更ではなく、集約全体を一括で更新します。

コレクション指向のリポジトリ

 さらに、「コレクション指向」のリポジトリについて細かく見ていきましょう。コレクション指向のリポジトリとは、コレクションクラス(java.util.HashSet)のように集約の追加/削除/取得ができます。

インターフェイスの設計

 コレクション指向のコードサンプル(JavaC#)を見てみましょう。次のコードにある通り、Add(追加)、Remove(削除)、Get(取得)といった基本的な操作が備わっていることがわかります。

[リスト]「コレクション指向リポジトリ」の公開インターフェイス(Java)
// ■ユーザー集約を管理するリポジトリのインターフェイス
public interface UserRepository {

    // ▼リポジトリに追加
    public void add(User aUser);

    // ▼リポジトリから削除
    public void remove(User aUser);

    // ▼名前でユーザー集約を取得
    public User userWithUsername(
            TenantId aTenantId,
            String aUsername);
~中略~
}

コレクション指向の実装クラス

 次に、実装コード(Java)を見てみましょう。O/Rマッパーを利用している「HibernateUserRepository」クラスです。Hibernateを利用して保存や削除を呼び出しています。また取得においても、クエリを組み立てて条件に合った集約を取り出していることがわかります。

[リスト]「コレクション指向リポジトリ」の実装クラス(Java)
// ■ユーザー集約を管理するリポジトリの実装クラス
public class HibernateUserRepository
        extends AbstractHibernateSession implements UserRepository {

    // ▼リポジトリに追加
    @Override
    public void add(User aUser) {
        try {
            this.session().saveOrUpdate(aUser);
        } catch (ConstraintViolationException e) {
            throw new IllegalStateException("User is not unique.", e);
        }
    }

    // ▼リポジトリから削除
    @Override
    public void remove(User aUser) {
        this.session().delete(aUser);
    }

    // ▼名前でユーザー集約を取得
    @Override
    public User userWithUsername( TenantId aTenantId,  String aUsername) {
        Query query = this.session().createQuery(
"from com.saasovation.identityaccess.domain.model.identity.User as _obj_ "
 + "where _obj_.tenantId = ? "
 + "and _obj_.username = ?");
        query.setParameter(0, aTenantId);
        query.setParameter(1, aUsername, Hibernate.STRING);
        return (User) query.uniqueResult();
    }
~中略~
}

次のページ
永続指向のリポジトリ

この記事は参考になりましたか?

  • X ポスト
  • このエントリーをはてなブックマークに追加
IDDD本から理解するドメイン駆動設計連載記事一覧

もっと読む

この記事の著者

WINGSプロジェクト 青木 淳夫(アオキ アツオ)

WINGSプロジェクトについて>有限会社 WINGSプロジェクトが運営する、テクニカル執筆コミュニティ(代表 山田祥寛)。主にWeb開発分野の書籍/記事執筆、翻訳、講演等を幅広く手がける。2018年11月時点での登録メンバは55名で、現在も執筆メンバを募集中。興味のある方は、どしどし応募頂きたい。著書記事多数。 RSS X: @WingsPro_info(公式)、@WingsPro_info/wings(メンバーリスト) Facebook

※プロフィールは、執筆時点、または直近の記事の寄稿時点での内容です

山田 祥寛(ヤマダ ヨシヒロ)

静岡県榛原町生まれ。一橋大学経済学部卒業後、NECにてシステム企画業務に携わるが、2003年4月に念願かなってフリーライターに転身。Microsoft MVP for Visual Studio and Development Technologies。執筆コミュニティ「WINGSプロジェクト」代表。主な著書に「独習シリーズ(Java・C#・Python・PHP・Ruby・JSP&サーブレットなど)」「速習シリーズ(ASP.NET Core・Vue.js・React・TypeScript・ECMAScript、Laravelなど)」「改訂3版JavaScript本格入門」「これからはじめるReact実践入門」「はじめてのAndroidアプリ開発 Kotlin編 」他、著書多数

※プロフィールは、執筆時点、または直近の記事の寄稿時点での内容です

この記事は参考になりましたか?

この記事をシェア

  • X ポスト
  • このエントリーをはてなブックマークに追加
CodeZine(コードジン)
https://codezine.jp/article/detail/11048 2018/09/10 14:00

おすすめ

アクセスランキング

アクセスランキング

イベント

CodeZine編集部では、現場で活躍するデベロッパーをスターにするためのカンファレンス「Developers Summit」や、エンジニアの生きざまをブーストするためのイベント「Developers Boost」など、さまざまなカンファレンスを企画・運営しています。

新規会員登録無料のご案内

  • ・全ての過去記事が閲覧できます
  • ・会員限定メルマガを受信できます

メールバックナンバー

アクセスランキング

アクセスランキング