第14章 JDBC API的最佳实践

14.1 基于Template的JDBC使用方式

Spring中的DataSource

Spring的数据访问框架在数据库资源的管理上全部采用JDBC2.0标准之后引入的javax.sql.DataSource接口作为标准,无论是JdbcTemplate还是各种ORM方案的集成,皆是如此。

鉴于DataSource的重要地位,我们有必要对其做进一步的了解。

1.DataSource的种类

DataSource的基本角色是ConnectionFactory,所有的数据库连接将通过DataSource接口统一管理。DataSource实现类根据功能强弱可以划分为以下三类。

简单的DataSource实现

这种DataSource实现通常只提供作为ConnectionFactory角色的基本功能,更多时候,我们会使用这类DataSource实现进行开发或者测试,而绝不会用于正式的生产环境。

这里不再介绍 有兴趣可以看原文

拥有连接缓冲池的DataSource实现

这一类 DataSource 实现,除了提供作为 ConnectionFactory 角色的基本功能之外,内部还会通过连接缓冲池对数据库连接进行管理,生产环境下的 DataSource 全都属于这一类。使用数据库连接缓冲池,可以在系统启动之初就初始化一定数量的数据库连接以备用,返回给客户端的 Connection 对象通过 close() 方法被关闭,实际上只是被返回给缓冲池,而不是真正的被关闭。 这极大地促进了数据库连接资源的复用,进而提高系统性能。

支持分布式事务的Datasource实现

确切地说,这一类的DataSource实现类应该是javax.sql.XADataSource的实现类,从XADataSource返回的数据库连接类型为javax.sql.XAConnection,而XAConnection扩展了javax.sql.PooleaConnection,所以也可以看出来,支持分布式事务的Datasource实现类,同样支持数据库连接的缓冲。

除非应用程序确实需要分布式事务,否则,我们没有必要使用这一类的DataSource实现,通常的拥有连接缓冲池的DataSource实现类就足够了。

2.DataSource的访问方式

我们可以在本地构造并持有相应的DataSource实现,也可以将相应的DataSource绑定到命名服务,这完全取决于当前应用程序的使用场景。不过,DataSource的位置将决定我们获取DataSource支持的不同方式。

本地DataSource访问

如果我们在当前应用程序的,上下文中构造并持有了相应的DataSource实现,那么,可以通过本地的DataSource引用对其进行访问。这将是最常用的方式,无论是在独立的应用程序中还是在应用服务器中。只要将相应的DataSource实现类所在的jar包加入应用程序的Classpath,我们就可以直接构造并访问它。而使用Spring的IoC容器来管理本地的DataSource资源将是最为理想的方式,只要在容器的配置文件中进行简单的配置即可,容器中任何需要该资源的对象都可以获得依赖注入。否则,或许得自己构建一个专门用于该DataSource存取的Singleton对象,以供相应对象调用。

现在,我们对“如何使用Spring的IoC容器配置和访问DataSource”已经非常熟悉了,所以废话少说,接着看下一种DataSource的访问方式如何?

远程DataSource访问

对于各种应用服务器提供的特有的DataSource实现,或者绑定到应用服务器命名服务的独立的DataSource实现,我们需要通过JNDI对其进行访问。对于运行于应用服务器的程序或者分布式应用来说,通过JNDI访问DataSource将是最为常见的方式。在Spring的IoC容器中,我们可以通过org.springframework.jndi.JndiObjectFactoryBean对这些DataSource进行访问,例如:

 
 
    <bean id="dataSource" class="org.springframework.jndi.JndiObjectFactoryBean">
    	<property name="jndiName">
    		<value>java:env/myDataSource</value>
    	</property>
    </bean>
 
 

JndiObjectFactoryBean是一个FactoryBean实现,容器中对其引用获得的将是get0bject()所返回的对象,而不是JndiObjectFactoryBean本身。在这里,当然就是我们要用的DataSource。

如果我们已经迁移到Spring2.x,并且使用的是基于XSD的配置格式,那么我们可以直接使用<jndi:lookup>来查找应用服务器上的DataSource,如下方代码所示。

 
    <jndi:lookup id="dataSource" jndi-name="java:env/myDataSource"/>
 
 

相对于通过直接指定JndiObjectFactoryBean来进行JNDI查找而言,基于XSD的方式显然简洁多了,而且所达到的效果是相同的。只不过,在使用前,不要忘记将jee的命名空间加入配置文件。

3.自定义DataSource实现

除了可以使用现有的各种DataSource实现,Spring在org.springframework.jdbc.datasource包下还提供了部分类帮助我们自定义实现相应的DataSource,前提是当前需求实在是比较特殊。

这里不再介绍 有兴趣可以看原文

JdbcDaoSupport

如果现在来实现一个DAO的话,肯定不会像原来那样使用底层的JDBC API来实现它了。最起码,我们会使用相应的DataSource提供数据库连接,使用JdbcTemplate进行数据库操作,基本上类似下方代码清单所演示的内容。

 
 
    public class GenericDao implements IDaoInterface {
        private DataSource dataSource;
        private JdbcTemplate jdbcTemplate;
 
        public GenericDao(DataSource ds, JdbcTemplate jt) {
            this.dataSource = ds;
            this.jdbcTemplate = jt;
        }
 
        public void update(DomainObject obj) {
            getJdbcTemplate().update(...);
        }
        ...
        // setter和getter方法定义
    }
 
 

单个的DAO实现这么做是没有问题的,但当存在多个类似的DAO实现的时候,我们就会考虑重构这些DAO实现类,将DataSource和JdbcTemplate全部提取到统一的超类中。

不过,这部分工作Spring已经为我们做了,它直接提供了org.springframework.jdbc.core.support.JdbcDaoSupport作为所有基于JDBC进行数据访问的DAO实现类的超类。所以,DAO直接继承JdbcDaoSupport就可以,如下所示:

 
 
    public class GenericDao extends JdbcDaoSupport implements IDaoInterface {
    	public void update(DomainObject ojb) {
      	getJdbcTemplate().update(...);
      }
      ...
      // setter和getter方法定义
    }
 
 

至此,基于JabeTemplate方式进行数据访问的所有内容就告一段落了,我们要进入下一单元——基于操作对象的JDBC使用方式。

14.2 基于操作对象的JDBC使用方式

Spring除了提供基于Template形式的JDBC使用方式,还对各种数据库操作以面向对象的形式进行建模,为我们使用JDBC进行数据访问提供了另一种视角。

在这种基于操作对象的JDBC使用方式中, 查询、更新、调用存储过程等数据访问操作,被抽象为操作对象,这些操作对象统一定义在org.springframework.jdbc.object包下,以org.springframework.jdbc.object.RdbmsOperation作为整个操作对象体系的顶层抽象定义(见图14-6)。

image-20220502184537405

RdbmsOperation是一个抽象类,它提供了所有子类所需要的公共设施,包括当前数据库操作对应的SQL语句的声明,参数列表处理,以及进行底层数据库操作所必需的JdbcTemplate实例等。所有的操作对象最终的数据访问都是通过JdbcTemplate进行的,这一点可以让我们清楚,实际上,基于操作对象的JDBC使用方式与基于JdbcTemplate的JDBC使用方式是统一的,只不过对待概念的视角上有所不同而已。

根据数据访问操作,RdbmsOperation分为三个主要分支,即查询操作对象分支、更新操作对象分支以及存储过程对象分支。

  • 查询操作对象分支。org.springframework.jdbc.object.SqlQuery抽象类是查询操作对象分支的“首脑”,它为子类提供了各种执行数据查询操作的方法实现,不过,查询结果的处理留给子类来实现。

  • 更新操作对象分支。org.springframework.jdbc.object.SqlUpdate是该分支的主要实现类,它是实体类,我们可以直接使用它进行数据库更新操作,也可以对其进行扩展,比如,在子类中提供基于强类型参数的更新方法定义。

  • 存储过程对象分支。在这个分支中,我们将更多地使用org.springframework.jdbc.object.StoredProcedure,而不是它的父类org.springframework.jdbc.object.SqlCall。因为SqlCall的主要作用是根据调用信息构建相应的CallableStatementCreator,StoredProcedure在SqlCall的基础上提供了执行调用存储过程的方法定义。

SqlQuery和SqlUpdate的父类是org.springframework.jdbc.object.SqlOperation,该类的主要工作是根据提供的SQL语句和参数列表信息,为子类sqlQuery和SqlUpdate提供相应的PreparedStatementCreator。

这样,SqlQuery和SqlUpdate直接根据父类提供的PreparedStatementCreator进行数据的查询和更新即可。

现在,让我们详细看一下每一个分支下,Spring具体提供了哪些可用的操作对象。

这里不做更多介绍 有兴趣可以看原文

14.3 小结

JDBC在Java平台上的数据访问领域一直占据重要地位,但在日常的开发过程中,糟糕的实践方式让JDBC的使用多少有些声名狼藉。Spring框架针对JDBCAPI在日常使用中暴露出来的种种问题,提供了一套有针对性的最佳实践方式,包括基于JdbcTemplate和基于操作对象的实践方式。

本章讲述了Spring框架提供的两种JDBCAPI的最佳实践方式,并详尽阐述了这两种最佳实践方式的实现原理和使用方式。不过,要访问数据,JDBC并非唯一的选择。在ORM如此盛行的年代,不提及Java平台上强盛的ORM家族,或许有些说不过去。接下来,我们就一起来看一下Spring框架是如何为现有的各种ORM解决方案提供封装和集成的。