文档章节

spring学习笔记(九)

casoc
 casoc
发布于 2014/02/19 16:41
字数 830
阅读 1437
收藏 10

前言

        前面已经总结了Spring JDBC 中使用JdbcTemplate的增删改操作,这篇博文继续总结JdbcTemplate的查询

查询数据

        与Hibernate相比,Spring JDBC的查询操作略显繁杂,但在许多需要快速查寻结果给出响应的网站如电商网站,使用Spring JDBC自己写出高效的SQL查询比Hibernate自动生成的SQL效率更佳,Spring JDBC提供的查询方法主要分为单值的查询和多值得查询,下面分别总结。

多值查询

        Spring JDBC提供了org.springframework.jdbc.core.RowCallbackHandler回调接口以及和RowCallbackHandler功能类似的RowMapper<T>接口,由于这两个接口都比较简单,就直接用例子来讲解

RowCallbackHandler

        例:通过User的id获取User对象

......
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的遍历

RowMapper<T>

        例:通过User的name查询User对象集合

......
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回收。

单值查询

        按照返回值得类型不用分为了不同的方法:

        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条数据事都会抛出异常

© 著作权归作者所有

casoc
粉丝 48
博文 85
码字总数 60735
作品 0
成都
程序员
私信 提问
加载中

评论(3)

casoc
casoc 博主
@dml124 我去试了一下发现返回值可以不用viod,猜测返回值也没有其他的特殊作用,分析如下因为设置了<aop:config proxy-target-class="true">,所以Spring底层会使用CGLIB的动态代理,CGLIB的动态代理视同过继承目标类,在子类中采用方法拦截的技术拦截父类的方法调用,同时通过修改方法中的字节码实现代码的织入,上面方法使用的增强,相当于在调用代理类的方法时,先调用父类的方法,再调用用afterMethod方法,虽然afterMethod方法有返回值,但没有赋值给任何变量,所以没有意义,上面的只是猜测,还是要看CGLIB和Sping AOP的源码才能确定
casoc
casoc 博主

引用来自“dml124”的评论

小伙子问你个问题,spring的aop 切面编程。如果想在切面方法不定义为void,比如定义为String 类型的。这个返回值有啥作用吗?<!-- AOP记录Action方法后日志 -->
  <bean id="aspect" class="com.agree.framework.aop.AspectAdvice">
    <property name="logService">
      <ref bean="LogService"/>
    </property>
  </bean>
  <aop:config proxy-target-class="true">
    <aop:pointcut id="adminAction" expression="execution(* com.agree.framework.web.action.administration.*.*(..))"/>
    <aop:aspect ref="aspect">
      <aop:after pointcut-ref="adminAction" method="afterMethod"/>
    </aop:aspect>
  </aop:config>
比如我的method=afterMethod 这个afterMethod方法能定义为String返回类型吗

这个我没有试过,白天有工作,晚上给你答复。
dml124
dml124
小伙子问你个问题,spring的aop 切面编程。如果想在切面方法不定义为void,比如定义为String 类型的。这个返回值有啥作用吗?<!-- AOP记录Action方法后日志 -->
  <bean id="aspect" class="com.agree.framework.aop.AspectAdvice">
    <property name="logService">
      <ref bean="LogService"/>
    </property>
  </bean>
  <aop:config proxy-target-class="true">
    <aop:pointcut id="adminAction" expression="execution(* com.agree.framework.web.action.administration.*.*(..))"/>
    <aop:aspect ref="aspect">
      <aop:after pointcut-ref="adminAction" method="afterMethod"/>
    </aop:aspect>
  </aop:config>
比如我的method=afterMethod 这个afterMethod方法能定义为String返回类型吗
我的RabbitMQ的学习成果

背景 在研发分布式事务的最终一致性事务模式时,使用了RabbitMQ。 在这之前也接触过RabbitMQ,但没有特别深入的去了解它的特性与原理。这次决定系统的学习一次,所以业余时间阅读大神们的书籍...

XuePeng77
04/15
266
0
ZHENFENGSHISAN/perfect-ssm

Quick Start 项目简介 ssm系列 ssm-demo:Spring+SpringMVC+Mybatis+easyUI整合 perfect-ssm:RESTful API+redis缓存 ssm-cluster:前后端分离+集群部署 ssm-dubbo:dubbo服务化 ssm-micro-se......

ZHENFENGSHISAN
2017/09/18
0
0
springmvc+mybatis学习笔记(汇总)

springmvc+mybatis学习笔记(汇总) 标签 : springmvc mybaits [TOC] 笔记分为两大部分:mybatis和springmvc mybatis springmvc 笔记内容主要是mybatis和springmvc的一些基本概念和使用方法,...

brianway
2016/03/30
2.3K
2
Spring Boot学习笔记

多模块开发 [SpringBoot学习]-IDEA创建Gradle多Module结构的SpringBoot项目 RabbitMQ RabbitMQ 安装 linux安装RabbitMQ详细教程 Ubuntu 16.04 RabbitMq 安装与运行(安装篇) ubantu安装...

OSC_fly
2018/07/26
0
0
springmvc学习笔记(6)-springmvc整合mybatis(IDEA中通过maven构建

springmvc学习笔记(6)-springmvc整合mybatis(IDEA中通过maven构建) 标签: springmvc mybatis [TOC] 本文主要展示如何在intellij IDEA中通过maven构建springmvc+mybatis框架的开发环境。 需求...

brianway
2016/03/10
507
0

没有更多内容

加载失败,请刷新页面

加载更多

Activity启动模式二

上篇文章Activity启动模式一主要介绍了Activity的四种启动模式,这些启动模式都是在AndroidManifest中进行配置的。除此之外,Android系统还通过Intent类提供了一些标志位,同样可以指定Activ...

ltlovezh
28分钟前
4
0
三原色还原

1、Color Filter Array — CFA 随着数码相机、手机的普及,CCD/CMOS 图像传感器近年来得到广泛的关注和应用。 图像传感器一般都采用一定的模式来采集图像数据,常用的有 BGR 模式和 CFA 模式...

天王盖地虎626
39分钟前
3
0
kubernetes pod exec接口调用

正文 一般生产环境上由于网络安全策略,大多数端口是不能为集群外部访问的。多个集群之间一般都是通过k8s的ApiServer组件提供的接口通信,如https://192.168.1.101:6443。所以在做云平台时,...

码农实战
今天
8
0
3_数组

3_数组

行者终成事
今天
8
0
经典系统设计面试题解析:如何设计TinyURL(二)

原文链接:https://www.educative.io/courses/grokking-the-system-design-interview/m2ygV4E81AR 编者注:本文以一道经典的系统设计面试题:《如何设计TinyURL》的参考答案和解析为例,帮助...

APEMESH
今天
7
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部