SHOEISHA iD

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

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

iBATIS.NETにてO/Rマッピングを行う

iBATIS.NETにてO/Rマッピングを行う(DAO Framework編)

iBATIS.NETを用いて多層アプリケーションを構築する


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

データベースを実装する際に「iBATIS.NET」を導入すると、プログラムからSQLを分離したり、パーシステンス層とサービス層を分離したりして、データベースの管理を簡単にすることができます。本稿では、このiBATIS.NETに含まれるの2つのフレームワークのひとつ「DAO Framework」について紹介します。

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

はじめに

 別稿「iBATIS.NETにてO/Rマッピングを行う(SQL Maps編)」に引き続き、本稿ではiBATIS.NETのDAO Framework機能について解説します。iBATIS.NETに関する概要およびインストール方法については、別稿の内容を参照してください。

対象読者

 .NETにて開発を行っている方、また基本的なSQLを理解している方を対象としています。

必要な環境

 サンプルはVisual Studio .NET 2003で作成し、.NET Framework 1.1で動作確認をしています。

サンプルアプリケーションについて

 本稿のサンプルは書籍マスタのメンテナンスを行うWindowsアプリケーションです。

書籍マスタのメンテナンス画面
書籍マスタのメンテナンス画面

 「SQL Maps」と「DAO Framework」を用いて、MDB(Microsoft Accessのデータベース形式)にアクセスを行い、書籍テーブルへの追加・更新・参照・削除といったデータベース操作を実行します。このサンプルにおけるiBATIS.NET・クラス・設定ファイルの関連図を示します。

アプリケーションの実装イメージ
アプリケーションの実装イメージ

 このアプリケーションはプレゼンテーション層(画面)・サービス層(ビジネスロジック)・パーシステンス層(データベースアクセス)というレイヤに分かれています。なおiBATIS.NET(DataMapper-1.2.1,DataAccess-1.6.1)がサンプルに同梱されているため、改めてiBATIS.NETを入手する必要はありません。

補足
 当サンプルは、C#のStandard版でも開くことができるように1つのEXEプロジェクトにて構成されていますが、本来は層別にDLLを作成してアクセス制御を行うことをお奨めします。なお、より本格的な実装に興味がある方はiBATISと同じサイトにて配布されているNPetStoreを参考にすると良いでしょう。NPetStoreはiBATISによるリファレンス的な実装で、多層のレイヤからなるASP.NETアプリケーションのプログラムです(ただし、公開されているNPetStoreのバージョンは最新ではない場合があるためiBATISのソース管理リポジトリも参考にすると良いでしょう)。

DAO(Data Access Object)導入のメリット

 本稿ではiBATIS.NETの「DAO Framework」について紹介します。MicrosoftがADO以前に提唱していたDAOを思い浮かべる方も多いと思いますが、iBATISのDAOはデータベースへのアクセス処理を行う「Data Access Object」パターンを意味し、以下のような問題を解決することができます。

  • サービス層とパーシステンス層の間の結合度を最小限にする。
  • 外部リソースに対するアクセスを1ヵ所にまとめる。
  • サービス層におけるプログラムの複雑さを最小限にする。

サービス層とパーシステンス層の間の結合度を最小限にする。

 パーシステンス層の中における実装クラスや例外クラスを隠すことが可能になります。例えばデータベースアクセスにADOを使用していれば「DataReader」「DataAdapter」「Connection」「Command」といったクラスを、SqlMapsを使用していれば「Mapper」「SqlMapper」といったクラスをサービス層に対して隠蔽します。そのためパーシステンス層内部の変更がサービス層へ与える影響が少なくなります。

外部リソースに対するアクセスを1ヵ所にまとめる。

 データベースのような外部リソースへのアクセスロジックがDAOに集中するため、OracleからSQL Serverへの移植といったRDBMSの変更、LDAPやWebServiceといった他のリソースの追加の場合にも修正箇所を局所化することができます。

サービス層におけるプログラムの複雑さを最小限にする。

 「DaoManager」クラスがファサード(サブシステムの共通APIを提供するパターン)となり、トランザクションの制御メソッドなどを提供します。サービス層からは、この「DaoManager」と「各DAOインターフェース」にアクセスするだけで良いため、プログラムがシンプルになります。

 このような品質や保守性の向上がはかれるため、中規模から大規模の開発の機会があれば「DAO Framework」の導入を検討してみると良いでしょう。

「DAO Framework」の実装

 それでは、「Book」クラスを例にして、パーシステンス層を実装する手順を解説します。

  1. IBookDAO」というインターフェースクラスを作成。
  2. BaseSqlMapDao」というDAOのスーパークラスを作成。
  3. BookSqlMapDao」というDAOの実装クラスを作成。
  4. 管理ファイル「dao.config」に「IBookDAO」と「BookSqlMapDao」を登録。

DAOのインターフェース「IBookDAO」(パーシステンス層)

 最初に、インターフェース「IBookDAO」を定義します。このインターフェースではInsertBookGetBookといったサービス層に公開するメソッドの定義を宣言します。

「IBookDAO.cs」
using System;
using IBatisSample.Domain.Books;

namespace IBatisSample.Persistence.Interface.Books
{
  public interface IBookDao
  {
    //登録処理の定義
    void InsertBook(Book book);
    //取得処理の定義
    Book GetBook(string bookIsbn);
   (中略)
  }
}

 サービス層から利用できるメソッドをインターフェースとして定義するのは、パーシステンス層とサービス層の結合度を下げるためです。インターフェースと会話することで依存関係が下がり「メンテナンス性の向上」「テストの容易性」といった品質の向上が図れるようになります。

DAOのスーパークラス「BaseSqlMapDao」(パーシステンス層)

 DAOの共通処理はスーパークラス「BaseSqlMapDao」に実装しています。

「BaseSqlMapDao.cs」一部抜粋
namespace IBatisSample.Persistence
{
  public class BaseSqlMapDao : IDao
  {
    //SqlMapperの取得
    protected SqlMapper GetLocalSqlMap()
    {
      DaoManager daoManager = DaoManager.GetInstance(this);
      SqlMapDaoSession sqlMapDaoSession =
        (SqlMapDaoSession)daoManager.LocalDaoSession;
      return sqlMapDaoSession.SqlMap;
    }  

    //Insertの実行
    protected object ExecuteInsert(
      string statementName, object parameterObject) 
    {
      SqlMapper sqlMap = GetLocalSqlMap();
      try 
      {
        return sqlMap.Insert(statementName, parameterObject);
      } 
      catch (Exception e) 
      {
        throw new IBatisNetException("INSERT実行エラー '"
          +statementName+"'  原因: " + e.Message, e);
      }
    }

    //SelectObject(1件)の実行    
    protected object ExecuteQueryForObject(
      string statementName, object parameterObject) 
    {
      SqlMapper sqlMap = GetLocalSqlMap();
      try 
      {
        return sqlMap.QueryForObject(statementName, parameterObject);
      } 
      catch (Exception e) 
      {
        throw new IBatisNetException("SELECT実行エラー '"
          +statementName+"'   原因: " + e.Message, e);
      }
    }    
    (中略)
  }
}

 このクラスでは、

  • セッションに応じたSqlMapper(別稿にて紹介)を取得。
  • SqlMapperを用いてINSERTSELECTの処理を実行。
  • DBアクセス時に発生しうる様々な例外処理をIBatisNetExceptionというDAOの共通例外としてラップ。

 といったDAOの共通処理を記述しています。(このクラスは必要に応じて拡張すると良いでしょう。)

DAOの実装クラス「BookSqlMapDao」(パーシステンス層)

 「IBookDao」インターフェースと「BaseSqlMapDao」クラスを継承して「BookSqlMapDao」クラスを作成します。「IBookDao」インターフェースにて定義したメソッドの実装を行う必要があります。

「BookSqlMapDao.cs」
using System;
using IBatisSample.Domain.Books;
using IBatisSample.Persistence;
using IBatisSample.Persistence.Interface.Books;

namespace IBatisSample.Persistence.MapperDao.Books
{
  public class BookSqlMapDao : BaseSqlMapDao, IBookDao
  {
    (中略)
    //書籍の登録
    public void InsertBook(Book book)
    {
      ExecuteInsert("InsertBook", book);
    }

    //書籍の取得
    public Book GetBook(int bookId)
    {
      //型の変換
      Book book = new Book();
      book.Isbn = bookIsbn;
      return (ExecuteQueryForObject("GetBookById",book) as Book);
    }
    (中略)
  }
}

 それぞれのメソッドからスーパークラス「BaseSqlMapDao」で定義したSQL実行コマンドを呼び出しています。「BaseSqlMapDao」を継承しているため、「SqlMapper」クラスの取得や例外の制御といった処理を記述する必要はありません。

管理ファイル「dao.config」の設定(パーシステンス層)

 「DAO Framework」の設定ファイルである「dao.config」の記述を行います。

  • 接続先データベースの設定
  • データベースへのアクセス方法の設定(ADO.NET/SQL Maps/NHibernate)
  • DAOのインタフェースと実装クラスの紐付け

 といった設定を行います。

接続先データベースの設定

 iBATISにて接続するデータベースの接続文字列を記述します。

「dao.config」抜粋
<properties resource="database.config"/>
<!-- DBの接続先 -->
<database>
  <provider name="OleDb1.1"/>            
  <dataSource name="Access" connectionString=
     "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=${datasource}"/>
</database>

 他のプロバイダにて接続したい場合には、OleDb1.1の設定をコメントし、SQL Serverなどのコメントを外して設定を有効にすると良いでしょう。なお「database.config」というプロパティファイルには「<add key="datasource" value="iBatisSample.mdb" />」と記述しているため${datasource}には「iBatisSample.mdb」がセットされます。

データベースへのアクセス方法の設定

 DBアクセスに使用するプロダクトを記述します。

「dao.config」抜粋
<!-- セッションハンドラの登録 -->
<daoSessionHandler id="SqlMap">
  <property name="resource" value="SqlMap.config"/>
</daoSessionHandler>

 「DAO Framework」は「ADO.NET」「SQL Maps」「NHibernate」の3つのDBアクセス方式をサポートしています。「SQL Maps」を使用する場合は、上記の記述で問題ありません。

DAOのインタフェースと実装クラスの紐付け

 DAOのインタフェースと実装クラスの紐付けの設定を行います。「interface」にインターフェース名を、「implementation」に実装クラス名を記述します。

「dao.config」抜粋
<daoFactory>
  <!-- DAOをここに追加していきます -->
  <dao interface="IBatisSample.Persistence.Interface.Books.
IBookDao,iBatisSample"
  implementation="IBatisSample.Persistence.MapperDao.Books.
BookSqlMapDao,iBatisSample"/>
  
  <dao interface="IBatisSample.Persistence.Interface.Hoge.
IHoge,IBatisSample"
  implementation="IBatisSample.Persistence.MapperDao.Hoge.
Hoge,IBatisSample"/>
</daoFactory>

 interfaceimplementationのペアは「完全修飾クラス名,アセンブリ名」の形式にて記述します。以前のインターフェースプログラムでは、Factoryパターンによって実装クラスをハードコーディングする必要がありましたが、最近はDI(Dependecy Injection)によって、設定ファイルの記述から実装クラスを設定することが一般的になりました。「DAO Framework」は、このような簡易のDI機能をサポートしているため、IBookDaoインターフェースにBookSqlMapDaoクラスを簡単に設定することができます。

 ここまでの設定で、パーシステント層の設定は完了です。

「BookService」クラスの実装(サービス層)

 最後に、DAOを利用するサービス層のプログラムを行います。

「BookService.cs」
using System;
using IBatisNet.DataAccess;
using IBatisSample.Persistence.Interface.Books;
using IBatisSample.Domain.Books;

namespace IBatisSample.Service
{
  public class BookService
  {

    private static BookService _instance = new BookService();
    private DaoManager _daoManager = null;
    private IBookDao _bookDao = null;
    
    //コンストラクタ
    public BookService()
    {
      //DaoManagerの取得
      _daoManager = ServiceConfig.GetInstance().DaoManager;
      //DaoManagerからBook用のDaoを取得
      _bookDao= _daoManager.GetDao(typeof(IBookDao)) as IBookDao;
    }

    // BookServiceクラスのインスタンスを取得
    public static BookService GetInstance() 
    {
      return _instance;
    }

    //書籍の取得
    public Book GetBook(int bookId) 
    {
      Book book = null;

      _daoManager.OpenConnection();
      book = _bookDao.GetBook(bookId);
      _daoManager.CloseConnection();

      return book;
    }

    //書籍の登録
    public void InsertBook(Book book) 
    {
      _daoManager.BeginTransaction();
      try
      {
        _bookDao.InsertBook(book);
        _daoManager.CommitTransaction();
      }
      catch
      {
        _daoManager.RollBackTransaction();
        throw;
      }
    }
    (中略)
  }
}

 サービス層である「BookService」クラスは、多数のクライアントの実行に耐えられるようにSingleton(負荷がかからないようにインスタンスを1つしかつくらないように制御するパターン)にて実装しています。また、「DaoManager」クラスは「ServiceConfig」というサポートクラスから取得しています。

 このサンプルプログラムの

「BookService.cs抜粋」
_daoManager = ServiceConfig.GetInstance().DaoManager;
_bookDao= _daoManager.GetDao(typeof(IBookDao)) as IBookDao;

_daoManager.BeginTransaction();
_bookDao.InsertBook(book);
_daoManager.CommitTransaction();

 という部分から、DAO Frameworkに沿った実装を行うことで

  • ファサードである「DaoManager」クラスから「DAOの取得」と「トランザクションの管理」が可能。
  • DAOである「IBookDao」のメソッドにて検索や更新処理の実施が可能。

 という点が理解できるかと思います。

 このように、「IBookDao」というインターフェースに「BookSqlMapDao」という実装クラスがDIできることが「DAO Framework」の重要な機能といえるでしょう。

各レイヤーの役割について

 今回のサンプルで紹介してきた「DAO Framework」は多層アプリケーションを前提にしています。設計や実装を行う場合には、以下のレイヤごとの役割を明確に理解しておくことが重要になります。

レイヤ役割使用パターン
プレゼンテーション層Form/Webによるデータの表示や入力。フォーマットチェック。なし
サービス層ビジネスロジックの実装。トランザクションの管理。Singlton/ファサード
パーシステンス層データベース(リソース)アクセス。CRUDの実装。DAO/DI/ファサード

 これらのレイヤの役割を理解した上で、「DAO Framework」をうまく利用すると、拡張や保守に強い多層アプリケーションを構築することができるでしょう。

iBATIS.NETのデバッグについて

 もしiBATIS.NETが正しく動作しない場合は、XMLの設定に誤りがある可能性が高いため、実行時のエラーメッセージを確認してください。log4netのログ出力メソッドの2つ目の引数に、キャッチした例外を設定することで詳細のログが見れます。

 原因判明が困難な場合はiBATIS.NETのソースコードをダウンロードし、参照設定をDLLではなくダウンロードしたプロジェクトに変更します。そしてメニューの[デバッグ]-[例外]のダイアログにて[例外がスローされた時]の設定を[デバッガで中断]に変更します。この設定をしておけば、エラー発生ポイントで実行が停止するので原因を追求しやすいでしょう。

まとめ

 iBATISの特長をまとめると以下のようになります。

  • 「DAO Framework」のDaoManagerによりサービス層のプログラムがシンプルになる。また、パーシステンス層とサービス層の間はインターフェースベースでの接続をしているので、パーシステンス層の変更や修正による影響を受けにくい。
  • 「SQL Maps」により、パーシステンス層の開発者はマッピングファイルにてSQLを集中管理することができる。また、サービス層・プレゼンテーション層の開発者はSQLを意識せずにオブジェクトによるアプリケーション構築に専念できる。
  • SQLはマッピングファイルに定義されているため、軽微なSQLの修正であれば、アプリケーションの修正が不要である。
  • 活発なオープンソースプロジェクトであるApacheのプロダクトであり、Javaにも存在するため、Java業務への応用が図れる可能性がある。

 iBATISには、以上のような長所がありますので、興味を持たれた場合には、マニュアル、チュートリアル、NPetStore、ソースコード付属のテストコードなどをご覧になってみてはいかがでしょうか。

参考資料

  1. Apache iBATIS(英語)
  2. 上級DAOプログラミング(developerWorks)
  3. Inversion of Control コンテナと Dependency Injection パターン(Martin Fowler)
  4. ドメインロジックとSQL(Martin Fowler)
  5. J2EEパターン~明暗を分ける設計の戦略(ピアソン・エデュケーション)
  6. .NETのアプリケーション アーキテクチャ(日経BPソフトプレス)
修正履歴

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

  • このエントリーをはてなブックマークに追加
iBATIS.NETにてO/Rマッピングを行う連載記事一覧
この記事の著者

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

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

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

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

この記事をシェア

  • このエントリーをはてなブックマークに追加
CodeZine(コードジン)
https://codezine.jp/article/detail/113 2006/10/11 19:33

おすすめ

アクセスランキング

アクセスランキング

イベント

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

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

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

メールバックナンバー

アクセスランキング

アクセスランキング