public class TasksJdbcDAO extends JdbcDaoSupportimplements TasksDAO {
public List getTasksNames() { JdbcTemplate jt = getJdbcTemplate(); return jt.query("select TASKNAME from tasks",new RowMapperResultReader(new TasksRowMapper())); } class TasksRowMapper implements RowMapper { public Object mapRow(ResultSet rs, int index) throws SQLException { return rs.getString(1); } } public static void main(String[] args)throws Exception { ApplicationContext ctx = new ClassPathXmlApplicationContext("SpringConfig.xml"); DataSource ds =(DataSource) ctx.getBean("dataSourceDBDirect"); TasksJdbcDAO taskDao = new TasksJdbcDAO();
taskDao.setDataSource(ds); Iterator tskIter = taskDao.getTasksNames().iterator(); while (tskIter.hasNext()) { System.out.println(tskIter.next().toString()); } }}
在上面的例子中,普通的和复杂的任务代码已经被移交到框架中。还应注意,借助于Spring JDBC,我们如何利用控制反转(IoC)容器来提供一种DataSource-我们仅把它注入到TasksJdbcDAO对象中。 控制反转背后的概念通常被表达为"不要找我,让我找你好了"。IoC把一些任务移交到了框架中,并且脱离出了应用程序代码。不是让你的代码调用一个传统的类库,而是一个IoC框架调用你的代码。存在于许多API中的生命周期回调,例如相应于会话EJB的setSessionContext()方法,正是展示了这种方法。 DataSource必须被注入到这个类中(或者其超类中),这是通过setDataSource()方法实现的。所有配置细节都远离了业务逻辑或客户端代码;这增加你的应用程序的松耦合性并因此而提高了程序的可测试性和可维护性。作为选择,我们还能在JNDI或servlet容器中建立一个DataSource,并用编程方式来检索它,然后把它注入到DAO对象中。下面是一个你可以使用的示例Spring bean配置文件-SpringConfig.xml: <?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN""http://www.springframework.org/dtd/spring-beans.dtd"><beans><bean id="dataSourceDBDirect"class="org.springframework.jdbc.datasource.DriverManagerDataSource"destroy-method="close"> <property name="driverClassName" value="oracle.jdbc.driver.OracleDriver"/> <property name="url" value="jdbc:oracle:thin:@localhost:1521:orcl"/> <property name="username" value="scott"/> <property name="password" value="tiger"/></bean></beans> 这个文件指示Spring bean容器实例化一个dataSourceDBDirect bean-它基于org.springframework.jdbc.datasource.DriverManagerDataSource类创建。 1) 基于Spring JDBC实现一个业务层 我们已经看到了一个简单的使用Spring JDBC的例子,在这种情况下,它从Spring BeanFactory(控制反转容器)中得到极少的帮助。现在,我们将超越这个简单的例子。让我们来探讨一下如何基于Spring JDBC实现业务服务。首先,让我们创建一个客户端-一个为终端用户提供输出的应用程序。该客户端使用了一个服务,一个遵守下面的Service接口的业务服务: package com.spring.jdbc;import java.util.List;public interface Service { public List getTasksNames(); public void setTasksDao(TasksDAO taskDAO);} 客户端需要存取一个业务服务对象。它将使用Spring BeanContainer来"抓住"这样的一个服务对象。客户端仅能针对接口编程并且依赖容器来提供一种实际的实现。而且,这个ServiceImpl类必须实现所有的存在于业务服务接口中的方法。该代码看上去如下所示: package com.spring.jdbc;import java.util.List;public class ServiceImpl implements Service{ TasksDAO taskDAO; public void setTasksDao(TasksDAO taskDAO) { this.taskDAO=taskDAO; } public List getTasksNames() { List tasks = taskDAO.getTasksNames(); return tasks; }} 你应该已经注意到,该服务需要一个TasksJdbcDAO。反过来,这个对象实现了TasksDAO接口。因此,我们将通过BeanFactory来把DAO注入到该服务中。在此,我们碰巧有一个TasksJdbcDAO类-bean工厂可以使用它来实现这一目的。然而,既然这个类派生于JdbcDaoSupport,那么我们知道我们需要注入一个DataSource或让bean工厂为我们注入该DataSource。现在这个bean配置文件看上去如下所示: <?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN""http://www.springframework.org/dtd/spring-beans.dtd"><beans><bean id="dataSourceDBDirect"class="org.springframework.jdbc.datasource.DriverManagerDataSource"destroy-method="close"> <property name="driverClassName" value="oracle.jdbc.driver.OracleDriver"/> <property name="url" value="jdbc:oracle:thin:@localhost:1521:orcl"/> <property name="username" value="scott"/> <property name="password" value="tiger"/></bean><bean id="tasksDAO" class="com.spring.jdbc.TasksJdbcDAO"> <property name="dataSource"> <ref local="dataSourceDBDirect"/> </property></bean><bean id="service" class="com.spring.jdbc.ServiceImpl"> <property name="tasksDao"> <ref local="tasksDAO"/> </property></bean></beans> 我们看到服务bean使得tasksDao bean被注入-它反过来又使dataSourceDBDirect对象被注入。当我们请求服务bean时,我们通过一个具有DataSource的DAO得到它。至此,一切就绪。因此,当客户端存取bean容器以得到服务对象时,会发生什么呢?该bean容器实例化并且注入一个DataSource和一个TasksDAO-在把服务返回到客户端之前。现在,我们的客户端变得相当简单了。它需要与BeanFactory进行通讯,"抓住"一个服务对象并处理它: package com.spring.jdbc;import java.util.Iterator;import org.springframework.context.ApplicationContext;import org.springframework.context.support.ClassPathXmlApplicationContext;public class Client extends RuntimeException { public static void main(String[] args) throws Exception { ApplicationContext ctx = new ClassPathXmlApplicationContext("SpringConfig.xml"); Service service = (Service)ctx.getBean("service"); Iterator tskIter = service.getTasksNames().iterator(); while (tskIter.hasNext()) { System.out.println(tskIter.next().toString()); } }} 你必须注意,在此Client派生于RuntimeException异常类。Spring抛出了RuntimeExceptions而不是检查的异常-RuntimeExceptions不应该被捕获。由于在你的代码中捕获所有异常是一种复杂的任务,所以Spring开发者决定抛出RuntimeExceptions以便实现如果你不捕获一个异常的话,那么你的应用程序将会中断而且用户会得到该应用程序异常。使用它们的第二个理由是,绝大多数异常都是不可恢复的,因此你的应用程序逻辑不能以任何方式来再次处理它们。 五. 另外的优点 除了上面描述的Spring框架带给JDBC中的优点外,与你的JDBC应用程序一起使用Spring框架还存在另外一些优点。这些优点包括: · Spring框架提供了 org.springframework.jdbc.support.nativejdbc.NativeJdbcExtractor接口和这个接口的一些实现(例如SimpleNativeJdbcExtractor)。对于经由一个Oracle连接或ResultSet访问Oracle特征的情况来说,这些内容是非常有用的。 · 对于创建oracle.sql.BLOB(二进制大型对象)和oracle.sql.CLOB(字符大型对象)实例来说,Spring提供了类org.springframework.jdbc.support.lob.OracleLobHandler。 · Spring提供的OracleSequenceMaxValueIncrementer类提供了一个Oracle序列的下一个值。它有效地提供了与你直接使用下列命令:"select someSequence.nextval from dual"(其中,someSequence是在Oracle数据库中的你的序列的名字)所提供的一样的信息。这种方法的优点是,DataFieldMaxValueIncrementer接口可以用于一个DAO层次中而不必紧密地耦合于Oracle特定的实现。 六. 结论 本文集中讨论了使用Spring来编写更可维护的和更不易出现错误的JDBC代码。Spring JDBC提供了一些优点,例如更为干净的代码,更好的异常与资源处理,并且能够集中于业务问题而不是复杂的任务代码。另外,值得注意的是,使用Spring框架能够使用极少的代码就可以实现实质上与传统型JDBC相同的功能。