3
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Doma入門 - Criteira APIで射影を使う

Last updated at Posted at 2020-10-07

はじめに

SELECT句で特定のカラムだけを指定して取得することを射影と呼びますが、全てのカラムを取得すると都合が悪い場合(カラムの数が多すぎる、LOBのような容量の大きなカラムが存在するなど)に便利な手法です。

今回は、Doma のCriteria APIで射影を表現する方法を説明します。

Criteria APIで射影を実現する方法は2つあります。

  • エンティティクラスを使って取得する
  • タプルクラスを使って取得する

ここで使うDomaのバージョンは3.6.0です。

DomaやCriteria APIの概要についてはDoma入門の他の記事もお読みください。

この記事で使うサンプルコード

簡略化したコードで説明します。より詳しいサンプルコードについては下記のプロジェクトを参照してください。

データベースには従業員を表すemployeeテーブルがあります。

schema.sql
create table employee (
    id integer not null primary key,
    name varchar(255) not null,
    age integer not null,
    salary integer,
    job_type varchar(20),
    hiredate timestamp, 
    department_id integer, 
    version integer not null, 
    insertTimestamp timestamp, 
    updateTimestamp timestamp);

employeeテーブルに対応するEmployeeクラスを用意します。

Employee.java
@Entity(metamodel = @Metamodel)
public class Employee {
  @Id
  Integer id;
  String name;
  Age age;
  Salary salary;
  @Column(name = "JOB_TYPE")
  JobType jobType;
  LocalDate hiredate;
  @Column(name = "DEPARTMENT_ID")
  Integer departmentId;
  @Version
  @Column(name = "VERSION")
  Integer version;
  LocalDateTime insertTimestamp;
  LocalDateTime updateTimestamp;

  // getter, setter
}

射影した結果を保持するDTO(ValueObject)を用意します。

NameAndSalaryDto.java
public class NameAndSalaryDto {

  private final String name;
  private final Salary salary;

  public NameAndSalaryDto(String name, Salary salary) {
    this.name = name;
    this.salary = salary;
  }

  // getter
}

EmployeeRepositoryを用意し、このクラスにいくつかメソッドを追加して例を示していきます。

EmployeeRepository.java
public class EmployeeRepository {

  private final QueryDsl queryDsl;

  public EmployeeRepository(Config config) {
    this.queryDsl = new QueryDsl(config);
  }
}

エンティティクラスを使って取得する

これはすでにあるデータ構造を再利用できると言う意味で非常にお手軽な方法です。

projectToもしくはselectToメソッドを使えば射影した値をエンティティの特定のプロパティにセットできます(その他のプロパティにはnullや初期値がセットされます)。
projectToメソッドは結果から重複したエンティティIDを削除しますが、selectToメソッドは削除しません。

  public List<Employee> selectNameAndSalaryAsEntityList() {
    Employee_ e = new Employee_();
    return queryDsl.from(e).selectTo(e, e.name, e.salary).fetch();
  }

発行されるSQLは次のものになります。

select t0_.id, t0_.name, t0_.salary from Employee t0_

nameとsalaryの2つのカラムがSELECT句に指定されたSQLになります。指定していない主キーが含まれていますが、これは仕様で、結果セット内においてエンティティの一意性を保証するために使われます。

タプルクラスを使って取得する

Domaは、org.seasar.doma.jdbc.criteria.tupleパッケージでTuple2Tuple3、...、Tuple9といった値の組み合わせを表すクラスを提供します。
クラス名の末尾の数字はそのクラスでいくつの値を扱えるかと言うことを表しています。

次はTuple2クラスを使って2つのカラムを射影する例です。

  public List<Tuple2<String, Salary>> selectNameAndSalary() {
    Employee_ e = new Employee_();
    return queryDsl.from(e).select(e.name, e.salary).fetch();
  }

発行されるSQLは次のものになります。

select t0_.name, t0_.salary from Employee t0_

nameとsalaryの2つのカラムのみがSELECT句に指定されたSQLになっています。

Tuple2〜9のクラスをどのレイヤーでどのように利用するかは自由ですが、以下のようにアプリケーションのDTO(もしくはValueObject)に変換するまでの一時的なデータ構造として使うのが綺麗な設計だと思います。

  public List<NameAndSalaryDto> selectNameAndSalaryAsNameAndSalaryDtoList() {
    Employee_ e = new Employee_();
    return queryDsl.from(e).select(e.name, e.salary).stream()
        .map(tuple -> new NameAndSalaryDto(tuple.getItem1(), tuple.getItem2()))
        .collect(Collectors.toList());
  }

10個以上のカラムを指定して受け取る場合はorg.seasar.doma.jdbc.criteria.tuple.Rowインタフェースで表現することもできますが、カラムの数が多い場合は読み取り専用のエンティティクラスを作成したほうがわかりやすいでしょう。

おわりに

Criteria APIで射影を表現する方法を説明しました。

TupleクラスのようなものがJavaの標準にあればそれを使いたいところですが、実際にはないのが悩ましいところですね。

なお、DomaのTuple2〜9クラスはKotlinで使われることも意識しているのでdestructuring declarationに対応しています。そのためKotlinでは次のように扱えます。

val (number, string) = Tuple2(1, "a")
3
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?