11.1 使用Spring JDBC访问数据库

2024-04-17 02:58

本文主要是介绍11.1 使用Spring JDBC访问数据库,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

参考《企业应用开发实战》第11章。

细节:
如果某参数不希望在方法中改变,可以声明为final

实战经验:
一般在配置文件中先声明一下 dataSource、jdbcTemplate等bean,之后在需要用的DAO类中,一般先写个基类BaseDao,在基类中定义一些通用的功能,例如声明JDBCTemplate、分页查询等多种。

@Autowired
private JdbcTemplate jdbcTemplate;

更改

jdbcTemplate的update语句:
比较好的写法是用 ?占位符

String sql = " INSERT INTO t_post(topicId,postText) VALUES(?,?)";               
Object[] params = new Object[]{post.getTopicId(),post.getPostText()};
jdbcTemplate.update(sql, params,new int[]{Types.VARCHAR,Types.VARCHAR});

将某个类的对象插到数据库中,想让主键值自动绑定到该对象上?方便后继的使用

//  ForumJdbcDao里的方法@Overridepublic void addForum(Forum forum) {// TODO Auto-generated method stubfinal String sql = "INSERT INTO t_forum(forumName,forumDesc) VALUES(?,?)";// 创建一个主键执有者KeyHolder keyHolder = new GeneratedKeyHolder();getJdbcTemplate().update(new PreparedStatementCreator() {public PreparedStatement createPreparedStatement(Connection conn)throws SQLException {PreparedStatement ps = conn.prepareStatement(sql);ps.setString(1, forum.getForumName());ps.setString(2, forum.getForumDesc());return ps;}}, keyHolder);// 从主键执有者中获取主键forum.setForumId(keyHolder.getKey().intValue());}

批量更改数据

//  一次性新增多个论坛版块
public void addForums(final List<Forum> forums) {final String sql = "INSERT INTO t_forum(forum_name,forum_desc) VALUES(?,?)";jdbcTemplate.batchUpdate(sql, new BatchPreparedStatementSetter() {public int getBatchSize() {return forums.size();}public void setValues(PreparedStatement ps, int index)throws SQLException {Forum forum = forums.get(index);ps.setString(1, forum.getForumName());ps.setString(2, forum.getForumDesc());}});}

查询

经过SQL语句查询得到的数据需要转化为类的对象并返回!
查询得到单条数据:
Spring会遍历查询的结果集,对结果集的每一行调用RowCallbackHandler回调接口处理数据
与RowCallbackHandler类似的RowMapper就不详说了(见书P371)

// 查询forum_name,forum_desc是为了后面返回Forum的对象
public Forum getForum(final int forumId) {String sql = "SELECT forum_name,forum_desc FROM t_forum WHERE forum_id=?";final Forum forum = new Forum();jdbcTemplate.query(sql, new Object[] { forumId },new RowCallbackHandler() {public void processRow(ResultSet rs) throws SQLException {forum.setForumId(forumId);forum.setForumName(rs.getString("forum_name"));forum.setForumDesc(rs.getString("forum_desc"));}});return forum;}

查询返回多条数据:

// 查询forum_id在一组区间内的多条数据
public List<Forum> getForums(final int fromId, final int toId) {String sql = "SELECT forum_id,forum_name,forum_desc FROM t_forum WHERE forum_id between ? and ?";List<Forum> forums = new ArrayList<Forum>();jdbcTemplate.query(sql,new Object[]{fromId,toId},newRowCallbackHandler(){ public void processRow(ResultSet rs) throws SQLException { Forum forum = new Forum();forum.setForumId(rs.getInt("forum_id"));forum.setForumName(rs.getString("forum_name"));forum.setForumDesc(rs.getString("forum_desc")); forums.add(forum);}}); return forums;}

查询单值数据

//1.查询一行数据并返回int型结果  
jdbcTemplate.queryForInt("select count(*) from test");  
//2. 查询一行数据并将该行数据转换为Map返回  
jdbcTemplate.queryForMap("select * from test where name='name5'");  
//3.查询一行任何类型的数据,最后一个参数指定返回结果类型  
jdbcTemplate.queryForObject("select count(*) from test", Integer.class);  
//4.查询一批数据,默认将每行数据转换为Map       
jdbcTemplate.queryForList("select * from test");  
//5.只查询一列数据列表,列类型是String类型,列名字是name  
jdbcTemplate.queryForList("  
select name from test where name=?", new Object[]{"name5"}, String.class);  
//6.查询一批数据,返回为SqlRowSet,类似于ResultSet,但不再绑定到连接上  
SqlRowSet rs = jdbcTemplate.queryForRowSet("select * from test");  

调用存储过程

具体见P375

先创建存储过程:
这里写图片描述

public int getUserTopicNum(final int userId) {// 调用存储过程String sql = "{call P_GET_TOPIC_NUM(?,?)}";Integer num = jdbcTemplate.execute(sql,new CallableStatementCallback<Integer>() {public Integer doInCallableStatement(CallableStatement cs)throws SQLException, DataAccessException {cs.setInt(1, userId); // 绑定传入参数cs.registerOutParameter(2, Types.INTEGER); //注册输出参数cs.execute();return cs.getInt(2);  //获取输出参数的值}});return num;}

BLOB/CLOB类型数据的操作

LOB代表大数据对象,其中BLOB用于存储大块的二进制数据例如图片、视频,CLOB用于存储长文本数据例如帖子。
数据库内部对应的类型是:Oracle对应BLOB/CLOB,MySQL对应的是BLOB/longtext。
longtext操作起来和简单类型一样,但是用户可能用流的方式操作LOB类型的数据。

写 BLOB/CLOB类型

例:
--- Post.java
包含多个属性,例如// 在数据库中postText是longtext(相当于CLOB),postAttach是BLOBprivate String postText;private byte[] postAttach;--- PostDao.javaimport java.sql.PreparedStatement;
import java.sql.SQLException;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.core.support.AbstractLobCreatingPreparedStatementCallback;
import org.springframework.jdbc.support.incrementer.DataFieldMaxValueIncrementer;
import org.springframework.jdbc.support.lob.LobCreator;
import org.springframework.jdbc.support.lob.LobHandler;
import org.springframework.stereotype.Repository;import chapter11.domain.Post;@Repository
public class PostDao {@Autowiredprivate JdbcTemplate jdbcTemplate;@Autowiredprivate LobHandler lobHandler;@Autowiredprivate DataFieldMaxValueIncrementer incre; public void addPost(final Post post){String sql = " INSERT INTO t_post(postId,userId,postText,postAttach)"+ " VALUES(?,?,?,?)";jdbcTemplate.execute(sql,new AbstractLobCreatingPreparedStatementCallback(this.lobHandler) {protected void setValues(PreparedStatement ps,LobCreator lobCreator)throws SQLException {//2:通过自增键指定主键值        ps.setInt(1, incre.nextIntValue());ps.setInt(2, post.getUserId()); // 设置CLOB和BLOB字段lobCreator.setClobAsString(ps, 3, post.getPostText());lobCreator.setBlobAsBytes(ps, 4, post.getPostAttach());}});}}注意点:一是类的标注 @Repository,二是主键自增的写法,三是CLOB和BLOB字段--- 配置文件<context:component-scan base-package="chapter11" /> <context:property-placeholder location="classpath:jdbc.properties" /><bean id="dataSource" class="org.apache.commons.dbcp2.BasicDataSource"destroy-method="close"p:driverClassName="${jdbc.driverClassName}"p:url="${jdbc.url}"p:username="${jdbc.username}"p:password="${jdbc.password}" /><bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"p:dataSource-ref="dataSource"/><bean id="namedParamJdbcTemplate"     class="org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate"><constructor-arg ref="dataSource"/></bean>      <bean id="nativeJdbcExtractor"      class="org.springframework.jdbc.support.nativejdbc.CommonsDbcpNativeJdbcExtractor"lazy-init="true" /><bean id="lobHandler" class="org.springframework.jdbc.support.lob.DefaultLobHandler"lazy-init="true" /><!-- 1:基于数据库序列的自增器 --><!-- <bean id="incre" class="org.springframework.jdbc.support.incrementer.OracleSequenceMaxValueIncrementer"p:incrementerName="seq_post_id"p:dataSource-ref="dataSource"/>  --><!-- 1:基于数据表的自增器 --> <bean id="incre"        class="org.springframework.jdbc.support.incrementer.MySQLMaxValueIncrementer"p:incrementerName="t_post_id"p:columnName="sequence_id"p:cacheSize="10"p:dataSource-ref="dataSource"/><bean id="transactionManager"       class="org.springframework.jdbc.datasource.DataSourceTransactionManager"p:dataSource-ref="dataSource"/><bean id="bbtForum" class="chapter11.service.JdbcBbtForum"p:forumDao-ref="forumDao"/>测试:
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"classpath:applicationContext.xml"})
@TransactionConfiguration
@Transactional
public class MyTest {@Autowiredprivate PostDao postDao;@Test@Rollback(false)public void testAddPost() throws Throwable{// 创建postPost post = new Post();post.setUserId(2);//  将图片转化为字节ClassPathResource res = new ClassPathResource("temp.jpg");byte[] mockImg = FileCopyUtils.copyToByteArray(res.getFile());post.setPostAttach(mockImg);post.setPostText("ceshi ");// 上面新建post时还没有对postId赋值默认初始值,在addPost方法中才对postId赋值postDao.addPost(post);}}
注意点:第一行的@RunWith(SpringJUnit4ClassRunner.class)容易丢导致出错

读BLOB、CLOB数据:
(1)以块数据方式读取Lob数据
以String读取Clob字段,以byte[]读取Blob字段

// 在PostDao类中加上方法
// 查看用户发表的所有帖子public List<Post> getAttachs(final int userId) {String sql = " SELECT post_id,post_attach FROM t_post where user_id =? and post_attach is not null ";return jdbcTemplate.query(sql, new Object[] { userId },new RowMapper<Post>() {public Post mapRow(ResultSet rs, int rowNum)throws SQLException {int postId = rs.getInt(1);byte[] attach = lobHandler.getBlobAsBytes(rs, 2);Post post = new Post();post.setPostId(postId);post.setPostAttach(attach);return post;}});}

(2)以流数据方式读取Lob数据
如果数据很大,可以用流数据方式读取

public void getAttach(final int postId, final OutputStream os){String sql = "SELECT post_attach FROM t_post WHERE post_id=? ";jdbcTemplate.query(sql, new Object[] {postId},new AbstractLobStreamingResultSetExtractor<Object>() {// 以流的方式处理LOB字段@Overrideprotected void streamData(ResultSet rs) throws SQLException, IOException, DataAccessException {// TODO Auto-generated method stubInputStream is = lobHandler.getBlobAsBinaryStream(rs, 1);if(is!=null)FileCopyUtils.copy(is, os);}protected void handleNoRowFound() throws LobRetrievalFailureException {System.out.println("Not Found result!");}});}

这篇关于11.1 使用Spring JDBC访问数据库的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/910617

相关文章

Spring Boot @RestControllerAdvice全局异常处理最佳实践

《SpringBoot@RestControllerAdvice全局异常处理最佳实践》本文详解SpringBoot中通过@RestControllerAdvice实现全局异常处理,强调代码复用、统... 目录前言一、为什么要使用全局异常处理?二、核心注解解析1. @RestControllerAdvice2

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

Python内置函数之classmethod函数使用详解

《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空