前面已经总结了Spring JDBC 中使用JdbcTemplate的增删改操做,这篇博文继续总结JdbcTemplate的查询
java
与Hibernate相比,Spring JDBC的查询操做略显繁杂,但在许多须要快速查寻结果给出响应的网站如电商网站,使用Spring JDBC本身写出高效的SQL查询比Hibernate自动生成的SQL效率更佳,Spring JDBC提供的查询方法主要分为单值的查询和多值得查询,下面分别总结。spring
Spring JDBC提供了org.springframework.jdbc.core.RowCallbackHandler回调接口以及和RowCallbackHandler功能相似的RowMapper<T>接口,因为这两个接口都比较简单,就直接用例子来说解
sql
例:经过User的id获取User对象
app
...... String sql = "SELECT u.name, u.age, u.phone, u.address FROM users.user u WHERE u.id=?"; //注意必须声明为final类型才能在后面的匿名内部类中使用 final User user = new User(); jdbcTemplate.query(sql, new Object[] {id}, new RowCallbackHandler() { public void processRow(ResultSet rs) throws SQLException { user.setId(id); user.setName(rs.getString("name")); user.setAge(rs.getInt("age")); user.setPhone(rs.getString("phone")); user.setAddress(rs.getString("address")); } }); ......
经过例子能够发现,和JDBC API的操做至关相似,只是省去建立链接,获取Statement,执行SQL最后关闭链接的过程。若查询出的结果集可能超过1条记录,只须要在query方法外声明一个final的对象List,在内部方法内部先new一个对象,再设值,最后加入List便可,processRow()方法会自动完成ResultSet的遍历网站
例:经过User的name查询User对象集合
ui
...... String sql = "SELECT u.name, u.age, u.phone, u.address FROM users.user u WHERE u.name=?"; List<User> users = jabcTemplate.query(sql, new Object[] {name}, new RowMapper<User>() { public User mapRow(ResultSet rs, int index) throws SQLException { User user = new User(); user.setId(rs.getString("id")); user.setName(name); user.setAge(rs.getInt("age")); user.setPhone(rs.getString("phone")); user.setAddress(rs.getString("address")); return user; } }) ......
RowMapper<T>使用了泛型,是内部方法能够直接返回咱们指定的类型,而且由Spring帮我自动生成对应的集合,感受上使用RowMapper<T>更加方便简单,可是RowMapper<T>在数据量很大时及其的消耗内存,好比说如今User表中有100万条数据,我想要给全部的用户发一封邮件,若是使用RowMapper<T>查询全部用户,Spring将把全部的用户初始化一个对象并放在List中,这个List将会很是的大,甚至直接形成OutOfMemoryException,及时没有你还须要遍历这个List,而后每遍历一次发一封邮件,遍历这么大的List想一想都以为恐怖。而若是使用RowCallbackHandler,你能够直接在内部方法中发邮件,咱们知道经过JDBC查询返回的结果集并不会一次将全部匹配数据返回,而是返回一部分,具体是多少由JDBC启动决定,当调用ResultSet#next()超过数据范围是才会去取下一批,这样能够有效的避免JVM内存开销过大,取了下一批后,上一批会有JAVA的GC回收。spa
按照返回值得类型不用分为了避免同的方法:
code
int queryForInt(String sql)
对象
int queryForInt(String sql, Object... args)接口
int queryForInt(String sql, Object[] args, int[] argTypes)
long queryForLong(String sql)
......
<T> T queryForObject(String sql, Class<T> requiredType)
......
单值查询在结果集为空或结果集超过1条数据事都会抛出异常