0
0

More than 1 year has passed since last update.

SQL取得時にネスト化されたリストオブジェクトに同時にマッピングする方法

Posted at

0.はじめに

ORマッパーとしてMybatisはメジャーと思いますが、これまであまり有効な使い方をしてこなかった方もいるかもしれません。
Java側から変数をバインドさせるとか、受け取ったDB取得結果をオブジェクトにマッピングするとか、if文で条件を可変にするとか。(私はその程度でした。。)
実は、ORマッパーはもっと複雑なこともマッピングしてくれて、その分楽になるんだよというような内容のご紹介になります。

1.概要

Model型のListかつ、Modelの中にもList型が内包されているようなリストの中にリストがネストされているオブジェクトの場合にDBから取得したデータを設定していく際には取得結果のForループに加えて、内包されているList型も設定していくためのForループが必要になる為、二重ループさせる必要があります。

もちろんこの実装でも誤りではありませんが、実装が増えることで可読性が下がったり、テストする際の確認ケースが増えたりコードが増えることによるバグのリスクがあがる可能性があるため、減らせるならば極力実装をシンプルにするほうが好ましいと思われます。

MyBatisのORマッパーにはネストされたオブジェクトに対してもマッピングすることができる機能があるため紹介させていただきます。

2.参考例

実際にどのようなケースで利用できるものなのかを以下の例をもとに紹介します。
商品とその商品コードに紐づく材料が複数件存在する場合に、複数のマスタからまとめて取得した内容をネスト化しているオブジェクトに設定したいケースです。

商品マスタ
商品コード 商品名
E001 レモンサワー
E002 スクリュードライバー
商品材料マスタ
No 商品材料コード 商品コード 商品名 材料価格(円)
1 E001-01 E001 焼酎 1000
2 E001-02 E001 炭酸水 50
3 E001-03 E001 レモン 100
4 E002-01 E002 ウォッカ 2000
5 E002-02 E002 オレンジジュース 100

2-1.取得させたいオブジェクト

商品Modelの項目にさらに商品材料Modelのリストをネストして持っている状態。

/**
 * 商品情報
 */
@Data
public class itemModel{
/** 商品名 **/
private Long itemName;
/** 商品コード **/
private Long itemCode;
/** 商品材料リスト **/
private List<itemMaterialModel> itemMaterialList;
}
/**
 * 商品材料情報
 */
@Data
public class  itemMaterialModel{
/** 商品材料名 **/
private Long itemMaterialName;
/** 商品材料コード **/
private String itemMaterialCode;
/** 材料価格 **/
private BigDecimal materialPrice;
}

2-2.商品情報取得イメージ

商品Modelリスト[0]
・商品名:レモンサワー
・商品コード:E001

商品材料リスト
[0]
・商品名:焼酎
・商品材料コード:E001-01
・価格:1000
[1]
・商品名:炭酸水
・商品材料コード:E001-02
・価格:50
[2]
・商品名:レモン
・商品材料コード:E001-03
・価格:100

商品Modelリスト[1]
・商品名:スクリュードライバー
・商品コード:E002

商品材料リスト
[0]
・商品名:ウォッカ
・商品材料コード:E002-01
・価格:2000
[1]
・商品名:オレンジジュース
・商品材料コード:E002-02
・価格:50

2-3.SQL

・取得したい要素はresalutMap内でキーとなるidとその他の項目情報をresultタグで指定します。
・collectionタグで囲っている部分が今回ネストしたい取得項目になります。これにより、指定したオブジェクトにネスト化したい要素をマッピングさせることができます。

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.example.demo.dao.ItemMapper">

    <resultMap id="ItemMap" type="com.example.demo.dto.itemMaterialModel">
        <id property="itemCode" column="itemCode"/>
        <result property="itemName" column="itemName" />
        <collection property="itemMaterialList" ofType="com.example.demo.dto.itemMaterialModel">
            <id property="itemMaterialCode" column="itemMaterialCode" />
            <result property="itemMaterialName" column="itemMaterialName" />
            <result property="materialPrice" column="materialPrice" />
        </collection>
    </resultMap>

<select id="search" resultMap="ItemMap">
SELECT 
 mi.itemName
 ,mi.itemCode
 ,mim.itemMaterialName
 ,mim.itemMaterialCode
 ,mim.materialPrice
FROM m_item mi
INNER JOIN m_item_material mim
  ON mim.itemCode = mi.itemCode
</select>

3.最後に

今回はネスト化されているオブジェクトにMybatisでマッピングさせるための実装のご紹介でした。

ただ実装するだけではなく、今使っているライブラリやフレームワークには実は実装そのものを楽にしたり、可読性を上げたり品質を保つための便利な機能があったりするので、そういったものを探してみるのも面白いかもしれませんね。

0
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
0
0