This post originated from an RSS feed registered with Java Buzz
by Brian McCallister.
Original Post: jDBI 2.12 and the SQL Object API
Feed Title: Waste of Time
Feed URL: http://kasparov.skife.org/blog/index.rss
Feed Description: A simple waste of time and weblog experiment
The latest release of jDBI, 2.12, includes a new set of APIs I have been mulling over for a couple of years now, ever since JDBC 4.0 dropped the “ease of development” features.
The sql object API lets you define annotated interfaces which generate all the needed rigamarole for you. Take, for example:
interfaceTheBasics{@SqlUpdate("insert into something (id, name) values (:id, :name)")intinsert(@BindBeanSomethingsomething);@SqlQuery("select id, name from something where id = :id")SomethingfindById(@Bind("id")longid);}
This snippet defines two methods, and annotates them with the SQL and how to bind the arguments into the generated prepared statements. Using them is equally easy,
In this case, we open an on-demand sql object. On demand means that it will obtain and release connections as needed, generally immediately before and after each method call. We then just call methods and we get database interactions.
We used a registered result set mapper here as well. This is also a new feature in 2.12, available on both the fluent api, and in the sql object api. Basically it just lets you register result set mappers which will be used to transform each row of the result set into some object, one to one. You can add an explicit mapper as well, via the @Mapper annotation, or by defining your own mapping annotation.
To get access to additional functionality, such as tranactions or access to the underlying handle, the sql object API has the idea of mixin interfaces. These are interfaces defined as part of the library which will be implemented on the sql object, for instance to use transactions with a sql object you would define your object as
interfaceUsesTransactionsextendsTransactional<UsesTransactions>{@SqlUpdate("insert into something (id, name) values (:id, :name)")voidinsert(@BindBeanSomethingsomething);@SqlUpdate("update something set name = :name where id = :id")intupdate(@BindBeanSomethings);@SqlQuery("select id, name from something where id = :it")SomethingfindById(@Bindintid);}
The Transactional interface defines begin(), commit(), rollback(), and checkpoint related friends, as well as a callback receiver which wraps the callback in a transaction:
The whole thing is designed to, hopefully, be fully tunable and extensable, in that you can define your own binding annotations, statement customizer annotations, and so on. It translates the interfaces into regular jDBI elements, so the various techniques folks already use for statement rewriting, externalizing sql, etc all work the same as before. Now, hopefully, we have just managed to reduce a whole lot of boilerplate, and make stubbing or mocking the databases a smidge easier!
The easiest way to get started using jDBI is through maven (or ivy, or sbt, everything supports the maven2 repo layout now) via org.jdbi:jdbi:2.12, or grabbing the jar if you aren’t using one of those.