百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

Spring JDBC-Spring对DAO的支持详细讲解

moboyou 2025-03-29 15:05 16 浏览

【摘要】 概述Spring的DAO理念统一的异常体系统一的数据访问模板 使用模板和回调机制模板类 数据源 配置数据源 DBCP数据源C3P0数据源 获取JNDI数据源Spr...

  • 概述
  • Spring的DAO理念
  • 统一的异常体系
  • 统一的数据访问模板使用模板和回调机制模板类
  • 数据源配置数据源DBCP数据源C3P0数据源获取JNDI数据源Spring的数据源实现类
  • 总结

概述

Spring对多个持久化技术提供了集成支持,包括Hibernate、MyBatis、JPA、JDO。 此外Spring还提供了一个简化JDBC API操作的Spring JDBC框架。

Spring面向DAO制定了一个通用的异常体系,屏蔽了持久化技术的异常,使业务层和具体的持久化技术实现解耦。

另外,Spring提供了模板类简化各种持久化技术的使用。

通用的异常体系和模板类是Spring整合各种持久化技术的不二法门。


Spring的DAO理念

DAO(DATA Acces Object)是用于访问数据的对象,虽然大多数情况下存储在数据库中,但是也可以存放在文件或者LDAP(轻量目录访问协议,Lightweight Directory Access Protocol)中。 DAO不但屏蔽了数据存储最重介质的不同,也屏蔽了具体的实现技术的不同。

早起,JDBC是主流选择,近些年,数据库持久化技术得到了长足的发展。 只要为数据访问定义好DAO接口,并使用具体的实现技术实现DAO接口的功能,就可以在不同的实现技术之间平滑的切换

我们来举个例子 : 如下是一个典型的DAO应用实例,在USerDAO中定义访问User数据对象的接口方法,业务层通过UserDao操作数据,并使用具体的持久化技术实现USerDao接口方法,这样就实现了业务层和具体的持久化技术之间的解耦

提供DAO抽象层的好处:

  • 首先可以很容易地构造模拟对象,方便单元测试的开展
  • 其次在使用切面会有更多的选择,可以使用JDK动态代理,又可以使用CGLib动态代理

Spring本质上希望以统一的方式整合底层的持久化技术,即以统一的方式进行调用及事务管理,避免让具体的实现侵入到业务层的代码中。 由于每种持久化技术都有各自的异常体系,所以Spring提供了统一的异常体系,使不同异常体系的阻抗得以消弭,方便定义出和具体实现技术无关的DAO接口,以及整合到相同的事务管理体系中。


统一的异常体系

统一的异常体系是整合不同的持久化技术的关键。 Spring提供了一套和实现技术无关的、面向DAO层语义的异常体系,并通过转换器将不同持久化技术的异常转换成Spring的异常

很多正统API或者框架中,检查型异常被过多的使用,以致在使用API时,代码中充斥了大量的try/cath样板式的代码。这样就导致一堆异常处理的代码喧宾夺主侵入到业务代码中,破坏了代码的整洁和优雅。

Spring在org.springframework.dao包中提供了一套完备优雅的DAO异常体系,
这些异常都继承于DataAccessException , 而DataAccessException本身又继承于NestedRuntimeException, NestedRuntimeException异常以嵌套的方式封装了源异常。 因此,虽然不同的持久化技术的特定异常被转换到Spring的DAO异常体系中,但原始的异常信息并不会丢失,我们依然可以通过getCaucse()方法获取原始的异常信息。

JDBC/Mybatis的异常转换器为
SQLErrorCodeSQLExceptionTranslator

Spring以分类手法建立了异常分类目录 ,如下所示(为第一层次的异常类,每个异常类下都可能有众多的异常类),一方面可以使开发者从底层细如针麻的技术细节中脱离出来,另一方面可以选择感兴趣的异常加以处理。


统一的数据访问模板

Spring为支持持久化技术分别提供了模板访问的方式,降低了使用各种持久化技术的难度,因此可以大幅度的提供开发效率。

使用模板和回调机制

我们先看一段使用JDBC进行数据操作的简单代码,已经尽可能的简化了整个处理的过程

public void saveUser(User user) {
        Connection connection = null;
        PreparedStatement stmt = null;

        try {
            // 获取连接
            connection = DriverManager.getConnection(url, user, password);

            // 启动事物
            connection.setAutoCommit(false);

            // 业务操作
            stmt = connection
                    .prepareStatement("insert into user(id ,name) values(?,?)");
            stmt.setLong(1, user.getUserId());
            stmt.setString(2, user.getUserName());

            // 执行提交
            stmt.execute();

            // 提交事务
            connection.commit();

        } catch (Exception e) {
            // 回滚
            connection.rollback();
        } finally {
            // 释放资源
            stmt.close();
            connection.close();
        }
    }
  
 

如上代码所述,JDBC访问数据按照如下流程

  • 获取连接
  • 开启事务(如果有需要)
  • 执行具体的数据访问操作
  • 提交/回滚事务
  • 关闭资源

我们可以看到只有具体的业务操作才是我们关心的, Spring将这些相同的数据访问流程固化到模板中,并将数据访问中固定和变化的部分分开,同时保证模板类是线程安全的,以便多个数据访问线程共享同一个模板实例。变化的部分通过回调接口开放出来,用于定义数据访问和结果返回的操作。 (如下图)

这样我们只需要编写回调接口,并调用模板类进行数据访问,就可以得到我们期待的结果:数据访问成功执行,前置和后置的样板化工作也按照顺序正确执行,在提供开发效率的同时保证了资源使用的正确性,彻底消除了因为忘记进行资源释放而引起的资源泄漏问题。


模板类

Spring为各种支持的持久化技术都提供了简化操作的模板和回调,在回调中编写具体的数据操作逻辑,使用模板执行数据操作,在Spring中这是典型的数据操作模式。

我们来了解下Spring为不同的持久化技术所提供的模板类

ORM持久化技术

模板类

JDBC/Mybatis

org.springframework.jdbc.core.JdbcTemplate

Hibernate

org.springframework.orm.hibernate3/4/5.HibernateTemplate

JPA

org.springframework.orm.jpa.JpaTemplate

JDO

org.springframework.orm.jdo.JdoTemplate

如果直接使用模板类,则演需要在DAP中定义一个模板对象并提供数据资源。 Spring为每种持久化技术都提供了支持列,支持类中已完成了这样的功能。 这样我们只需要扩展这些支持类,就可以直接编写实际的数据访问逻辑,因此更加方便。

ORM持久化技术

支持类

JDBC/Mybatis

org.springframework.jdbc.core.JdbcDaoSupport

Hibernate

org.springframework.orm.hibernate3/4/5.HibernateDaoSupport

JPA

org.springframework.orm.jpa.JpaDaoSupport

JDO

org.springframework.orm.jdo.JdoDaoSupport

这些类都是继承于dao.support.DaoSupport类, DaoSupport实现了InitializingBean接口,在afterPropertiesSet()接口中检查模板对象和数据源是否被正确设置,否则将抛出异常。

所有的支持类都是abstract,其目的是希望被继承使用,而非直接使用


数据源

在Spring中,不但可以通过JNDI获取应用服务器的数据源,也可以在Spring容器中配置数据源。 此外还可以通过代码的方式创建一个数据源,以便进行无容器依赖的单元测试。

配置数据源

Spring在第三方依赖包中包含了2个数据源的实现类包

  • Apache的DBCP
  • C2P0

我们可以在Spring配置文件中利用二者中的任何一个配置数据源。

DBCP数据源

因篇幅原因,另开一篇 Apache-DBCP数据库连接池解读

简略配置如下:


  
 

BasicDataSource提供了close()方法关闭数据源,所以必须设定destroy-method=”close”属性,以便Spring容器关闭时,数据源能够正常关闭。

假设数据库为MySQL,如果配置不当,会发生经典的“8小时为” 。

原因是MySQL在默认情况下发现一个连接空闲时间超过8小时,则会在数据库端自动关闭这个连接。 而数据源并不知道这个连接已经被数据库关闭了,当它将这个无用的连接返回个某个DAO时,DAO就会抛出无法获取Connection的异常。

如果采用DBCP默认配置,由于testOnBorrow默认为true,数据源在将连接交给DAO之前,会事先检查这个连接是否良好,如果连接有问题(在数据库端被关闭),则回取一个其他的连接给DAP,并不会有“8小时问题”。 但是这样每次都检查有效性,在高并发的应用中会带来性能问题。

一种推荐的高效方式是: 将testOnBorrow设置为false,而将testWhielIdel设置为true,再设置好
timeBetweenEvictionRunsMillis的值。 这样DBCP将通过一个后台线程定时的对空闲连接进行检测,当发现无用的空闲连接(那些被数据库关闭的连接)时,就会将它们清掉,只要将
timeBetweenEvictionRunsMillis设置为小于8小时,那些被MySQL关闭的空闲连接就可以被清除出去,从而避免“8小时问题”

当然,MySQL本身可以通过调整interactive-timeout(以秒为单位)配置参数,更改空闲连接的过期时间, 所以在设置
timeBetweenEvictionRunsMillis值时,必须首先获知MySQL空闲连接的最大过期时间。


C3P0数据源

因篇幅原因,另开一篇 C3P0-数据库连接池解读

一个简单的配置

     
         
        oracle.jdbc.driver.OracleDriver     
         
                    
        jdbc:oracle:thin:@ip:port:instanceName     
          
         
        artisan     
         
         
        artisan     
         
 
  
 

获取JNDI数据源

Java Naming and Directory Interface (Java命名和目录服务接口)

如果应用配置在高性能的应用服务器比如weblogic/websphere等,则可能希望使用应用本身提供的数据源。 应用服务器的数据源使用JNDI开放调用者使用,Spring为此专门提供了引用JNDI数据源的JndiObjectFactoryBean,我们来看一个简单的配置


    

  
 

通过jndiName指定引用的JNDI数据源名称

Spring为JavaEE资源提供了一个jee命名空间,通过jee命名空间,可以有效的简化Java EE资源的引用, 如下所示:

         
         
   
  
 

Spring的数据源实现类

Spring本身也提供了一个简单的数据源实现类
org.springframework.jdbc.datasource.DriverManagerDataSource

这个类实现了javax.sql.DataSource接口, 但 它并没有提供池化连接的机制,每次调用getConnection()获取新连接时,只是简单地创建一个新的连接。

因此,这个数据源类比较适合在单元测试 或简单的独立应用中使用,因为它不需要额外的依赖类。

下面,我们来看一下DriverManagerDataSource的简单使用。

DriverManagerDataSource ds = new DriverManagerDataSource ();         
ds.setDriverClassName("com.mysql.jdbc.Driver");         
ds.setUrl("jdbc:mysql://localhost:3309/sampledb");         
ds.setUsername("artisan");         
ds.setPassword("123456");         
Connection actualCon = ds.getConnection();   
  
 

当然,我们也可以通过配置的方式直接使用DriverManagerDataSource。 如下

 
   
      
      
      
      
   
  
 

总结

不管采用何种持久化技术,都需要定义数据源。Spring附带了两个数据源的实现类包,你可以自行选择进行定义。 在实际部署时,我们可能会直接采用应用服 务器本身提供的数据源, 这时,则可以通过JndiObjectFactoryBean或jee命名空间引用JNDI中的数据源

如果感觉小编写得不错,请素质三连:点赞+转发+关注。我会努力写出更好的作品分享给大家。更多JAVA进阶学习资料小编已打包好,可以关注私信找我领取哦!

相关推荐

Excel技巧:SHEETSNA函数一键提取所有工作表名称批量生产目录

首先介绍一下此函数:SHEETSNAME函数用于获取工作表的名称,有三个可选参数。语法:=SHEETSNAME([参照区域],[结果方向],[工作表范围])(参照区域,可选。给出参照,只返回参照单元格...

Excel HOUR函数:“小时”提取器_excel+hour函数提取器怎么用

一、函数概述HOUR函数是Excel中用于提取时间值小时部分的日期时间函数,返回0(12:00AM)到23(11:00PM)之间的整数。该函数在时间数据分析、考勤统计、日程安排等场景中应用广泛。语...

Filter+Search信息管理不再难|多条件|模糊查找|Excel函数应用

原创版权所有介绍一个信息管理系统,要求可以实现:多条件、模糊查找,手动输入的内容能去空格。先看效果,如下图动画演示这样的一个效果要怎样实现呢?本文所用函数有Filter和Search。先用filter...

FILTER函数介绍及经典用法12:FILTER+切片器的应用

EXCEL函数技巧:FILTER经典用法12。FILTER+切片器制作筛选按钮。FILTER的函数的经典用法12是用FILTER的函数和切片器制作一个筛选按钮。像左边的原始数据,右边想要制作一...

office办公应用网站推荐_office办公软件大全

以下是针对Office办公应用(Word/Excel/PPT等)的免费学习网站推荐,涵盖官方教程、综合平台及垂直领域资源,适合不同学习需求:一、官方权威资源1.微软Office官方培训...

WPS/Excel职场办公最常用的60个函数大全(含卡片),效率翻倍!

办公最常用的60个函数大全:从入门到精通,效率翻倍!在职场中,WPS/Excel几乎是每个人都离不开的工具,而函数则是其灵魂。掌握常用的函数,不仅能大幅提升工作效率,还能让你在数据处理、报表分析、自动...

收藏|查找神器Xlookup全集|一篇就够|Excel函数|图解教程

原创版权所有全程图解,方便阅读,内容比较多,请先收藏!Xlookup是Vlookup的升级函数,解决了Vlookup的所有缺点,可以完全取代Vlookup,学完本文后你将可以应对所有的查找难题,内容...

批量查询快递总耗时?用Excel这个公式,自动计算揽收到签收天数

批量查询快递总耗时?用Excel这个公式,自动计算揽收到签收天数在电商运营、物流对账等工作中,经常需要统计快递“揽收到签收”的耗时——比如判断某快递公司是否符合“3天内送达”的服务承...

Excel函数公式教程(490个实例详解)

Excel函数公式教程(490个实例详解)管理层的财务人员为什么那么厉害?就是因为他们精通excel技能!财务人员在日常工作中,经常会用到Excel财务函数公式,比如财务报表分析、工资核算、库存管理等...

Excel(WPS表格)Tocol函数应用技巧案例解读,建议收藏备用!

工作中,经常需要从多个单元格区域中提取唯一值,如体育赛事报名信息中提取唯一的参赛者信息等,此时如果复制粘贴然后去重,效率就会很低。如果能合理利用Tocol函数,将会极大地提高工作效率。一、功能及语法结...

Excel中的SCAN函数公式,把计算过程理清,你就会了

Excel新版本里面,除了出现非常好用的xlookup,Filter公式之外,还更新一批自定义函数,可以像写代码一样写公式其中SCAN函数公式,也非常强大,它是一个循环函数,今天来了解这个函数公式的计...

Excel(WPS表格)中多列去重就用Tocol+Unique组合函数,简单高效

在数据的分析和处理中,“去重”一直是绕不开的话题,如果单列去重,可以使用Unique函数完成,如果多列去重,如下图:从数据信息中可以看到,每位参赛者参加了多项运动,如果想知道去重后的参赛者有多少人,该...

Excel(WPS表格)函数Groupby,聚合统计,快速提高效率!

在前期的内容中,我们讲了很多的统计函数,如Sum系列、Average系列、Count系列、Rank系列等等……但如果用一个函数实现类似数据透视表的功能,就必须用Groupby函数,按指定字段进行聚合汇...

Excel新版本,IFS函数公式,太强大了!

我们举一个工作实例,现在需要计算业务员的奖励数据,右边是公司的奖励标准:在新版本的函数公式出来之前,我们需要使用IF函数公式来解决1、IF函数公式IF函数公式由三个参数组成,IF(判断条件,对的时候返...

Excel不用函数公式数据透视表,1秒完成多列项目汇总统计

如何将这里的多组数据进行汇总统计?每组数据当中一列是不同菜品,另一列就是该菜品的销售数量。如何进行汇总统计得到所有的菜品销售数量的求和、技术、平均、最大、最小值等数据?不用函数公式和数据透视表,一秒就...