LoginSignup
0
0

More than 5 years have passed since last update.

JDBIでOneToMany

Last updated at Posted at 2017-03-12

一応 stackoverflow にサンプルはある。

この例だと、

  • もともとのクエリのメソッドを_getAccountByIdにして、新しく作ったgetAccountByIdを呼び出す
  • getAccountByIdでは1件だけ返すように調整。ない場合はnullを返す

のようにしている。

まとめると、Mapperが全行について呼ばれてしまう(_getAccountByIdの戻り値の型がリストになっていることに注目)ため、別のメソッドを生やして1件だけ返すようにしている。

しかし、上記の例だと、one側のidが1つの場合しか対応出来ないので、自分で別の例を書いてみた。


orderorder_detailがあり、1対nで対応しているとする。

-- create
create table `order` (id bigint);
create table order_detail (id bigint, order_id bigint);

-- input data
insert into `order` values (1);
insert into `order` values (2);
insert into order_detail values (1, 1);
insert into order_detail values (2, 1);
insert into order_detail values (3, 2);
// Groovy Version: 2.4.5 JVM: 1.8.0_77 Vendor: Oracle Corporation OS: Mac OS X
@GrabConfig(systemClassLoader = true)
@Grab('mysql:mysql-connector-java:5.1.31')
@Grab('org.jdbi:jdbi:2.78')

import groovy.transform.ToString
import org.skife.jdbi.v2.DBI
import org.skife.jdbi.v2.StatementContext
import org.skife.jdbi.v2.sqlobject.Bind
import org.skife.jdbi.v2.sqlobject.SqlQuery
import org.skife.jdbi.v2.sqlobject.customizers.Mapper
import org.skife.jdbi.v2.tweak.ResultSetMapper

import java.sql.ResultSet
import java.sql.SQLException
import java.util.concurrent.ConcurrentHashMap
import java.util.stream.Collectors

abstract class OrderDAO {
    @Mapper(OrderMapper.class)
    @SqlQuery("select o.id, od.id, od.order_id from `order` o join order_detail od on (o.id = od.order_id) where o.id = :id")
    abstract List<Order> _findOne(@Bind("id") long id)

    @Mapper(OrderMapper.class)
    @SqlQuery("select o.id, od.id, od.order_id from `order` o join order_detail od on (o.id = od.order_id)")
    abstract List<Order> _findAll()

    Order findOne(long id) {
        List<Order> orders = _findOne(id)
        return orders.isEmpty() ? null : orders.get(0)
    }

    List<Order> findAll() {
        List<Order> orders = _findAll()
        // 重複判定はもう少しちゃんとした方がよさそう
        return orders.stream().distinct().collect(Collectors.toList())
    }
}

@ToString
class Order {
    long id
    List<OrderDetail> details = new ArrayList<>()
}

@ToString
class OrderDetail {
    long id
    long orderId
}

class OrderMapper implements ResultSetMapper<Order> {
    private Map<Long, Order> orderMap = new ConcurrentHashMap<>()

    @Override
    Order map(int index, ResultSet r, StatementContext ctx) throws SQLException {
        long id = r.getLong("o.id")
        Order order = orderMap.get(id)
        if (order == null) {
            order = new Order()
            order.id = r.getLong("o.id")
        }
        OrderDetail detail = new OrderDetail()
        detail.id = r.getLong("od.id")
        detail.orderId = r.getLong("od.order_id")
        order.details.add(detail)
        orderMap.put(id, order)

        return order
    }
}

DBI dbi = new DBI("jdbc:mysql://localhost/sandbox", "root", "")

OrderDAO dao = dbi.onDemand(OrderDAO.class)
println dao.findOne(1L)
println dao.findOne(9L)
println dao.findAll()

結果

Order(1, [OrderDetail(1, 1), OrderDetail(2, 1)])
null
[Order(1, [OrderDetail(1, 1), OrderDetail(2, 1)]), Order(2, [OrderDetail(3, 2)])]

ps.

Mapperが実際に呼び出されている深いところまでは追っていないが、一応スレッドセーフにはなってるっぽい。
GParsで同時実行しながら、Mapperの中でランダムな文字列吐いて確認しても、全ての呼び出しで同じ文字が出てくる。が、結果は正しかった。
cglibがJDBI内にあるので、1つだけ生成して、あとは呼び出し時にそれを雛形に複製して割り当てているのかしら???

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