• 概述
  • 示例
    • BeanPropertySqlParameterSource 使用示例
    • MapSqlParameterSource使用示例
  • NamedParameterJdbcTemplate 支持 in 的操作
    • PrepareStatement的缺陷
    • NamedParameterJdbcTemplate的操作示例
  • 示例源码

概述

除了标准的JdbcTemplate外,Spring还提供了两个易用的JDBC模板类

  • SimpleJdbcTemplate 封装了JdbcTemplate,将常用的API开放出来 . 这里暂不讨论
  • NamedParameterJdbcTemplate 提供命名参数绑定的功能。

在低版本的Spring 中, 用户只能使用“?”占位符声明参数,并使用索引号绑定参数,必须要保证参数的索引号和SQL语句中的占位符“?”的位置正确匹配。

NamedParameterJdbcTemplate模板了支持命名参数变量的SQL,位于org.springframework.jdbc.namedparam包中,该包中还定义了一个用于承载命名参数的SqlParameterSource接口

  • BeanPropertySqlParameterSource:该实现类是将一个JavaBean对像封装成一个参数源,以便通过JavaBean属性名和SQL语句中的命名参数匹配的方式绑定参数

  • MapSqlparameterSource:该实现类内部通过一个Map存储参数,可以通过addValue(String paramName , Object value) 或 addValue(Map value)添加参数,并通过参数键名和SQL语句的命名参数的方式绑定参数。


示例

BeanPropertySqlParameterSource 使用示例

package com.xgj.dao.namedParameterJdbcTemplate.dao;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.namedparam.BeanPropertySqlParameterSource;
import org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate;
import org.springframework.jdbc.core.namedparam.SqlParameterSource;
import org.springframework.stereotype.Repository;import com.xgj.dao.namedParameterJdbcTemplate.domain.Artisan;/*** * * @ClassName: ArtisanNJDaoImpl* * @Description: @Repository标注的DAO层,受Spring管理* * @author: Mr.Yang* * @date: 2017年9月30日 上午12:42:26*/@Repository
public class ArtisanNJDaoImpl implements ArtisanNJDao {private NamedParameterJdbcTemplate namedParameterJdbcTemplate;private final static String insertArtisanSql = "insert into artisan(artisan_name) values(:artisanName)";/*** * * @Title: setNamedParameterJdbcTemplate* * @Description: 自动注入namedParameterJdbcTemplate* * @param namedParameterJdbcTemplate* * @return: void*/@Autowiredpublic void setNamedParameterJdbcTemplate(NamedParameterJdbcTemplate namedParameterJdbcTemplate) {this.namedParameterJdbcTemplate = namedParameterJdbcTemplate;}@Overridepublic void addArtisan(Artisan artisan) {// 定义命名参数SqlParameterSource sps = new BeanPropertySqlParameterSource(artisan);// 使用模板类方法namedParameterJdbcTemplate.update(insertArtisanSql, sps);}}

在SQL语句中声明命名参数的格式为

:paranName

比如values(:artisanName) ,多个参数使用逗号分隔。

在这个示例中,使用BeanPropertySqlParameterSource提供数据源,它接收一个JavaBean作为构造函数的入参,调用namedParameterJdbcTemplate.update(insertArtisanSql, sps)执行插入数据的而操作。

配置文件

<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"><!-- 扫描类包,将标注Spring注解的类自动转化Bean,同时完成Bean的注入 --><context:component-scan base-package="com.xgj.dao.namedParameterJdbcTemplate" /><!-- 使用context命名空间 配置数据库的properties文件 --><context:property-placeholder location="classpath:spring/jdbc.properties" /><bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"destroy-method="close" p:driverClassName="${jdbc.driverClassName}"p:url="${jdbc.url}" p:username="${jdbc.username}" p:password="${jdbc.password}" /><!-- 定义 namedParameterJdbcTemplate--><bean id="namedParameterJdbcTemplate" class="org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate"><constructor-arg ref="dataSource"/></bean></beans>

注意NamedParameterJdbcTemplate的配置,使用构造函数。


Domain

package com.xgj.dao.namedParameterJdbcTemplate.domain;import java.io.Serializable;public class Artisan implements Serializable {private static final long serialVersionUID = 1L;private String artisanId;private String artisanName;public String getArtisanId() {return artisanId;}public void setArtisanId(String artisanId) {this.artisanId = artisanId;}public String getArtisanName() {return artisanName;}public void setArtisanName(String artisanName) {this.artisanName = artisanName;}}

Artisan拥有两个属性,我们这里没有插入ID,暂且忽略。 其中 artisanName 这个属性和 SQL语句中的命名参数匹配,参数即按照这个匹配关系进行绑定。


单元测试

package com.xgj.dao.namedParameterJdbcTemplate.dao;import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.context.support.ClassPathXmlApplicationContext;import com.xgj.dao.namedParameterJdbcTemplate.domain.Artisan;public class ArtisanNJDaoImplTest {ClassPathXmlApplicationContext ctx = null;ArtisanNJDaoImpl artisanNJDaoImpl = null;@Beforepublic void initContext() {// 启动Spring 容器ctx = new ClassPathXmlApplicationContext("classpath:com/xgj/dao/namedParameterJdbcTemplate/conf_namedParameterJdbcTemplate.xml");artisanNJDaoImpl = ctx.getBean("artisanNJDaoImpl",ArtisanNJDaoImpl.class);System.out.println("initContext successfully");}@Testpublic void queryTeacherById() {Artisan artisan = new Artisan();artisan.setArtisanName("ArtisanNJ");artisanNJDaoImpl.addArtisan(artisan);}@Afterpublic void closeContext() {if (ctx != null) {ctx.close();}System.out.println("close context successfully");}
}

结果:
查看下数据库是否插入成功 (只是演示,忽略ID…)


MapSqlParameterSource使用示例

如果有数据表记录没有对应的领域对象,则用户可以直接使用MapSqlparameterSource达到绑定参数的目的。

public void addArtisanWithMapSqlParameterSource(Artisan artisan) {// 使用MapSqlParameterSource绑定参数MapSqlParameterSource mapSqlParameterSource = new MapSqlParameterSource().addValue("artisanName", artisan.getArtisanName());// 使用模板类方法namedParameterJdbcTemplate.update(insertArtisanSql,mapSqlParameterSource);}

由于MapSqlParameterSource中的大多数方法都能返回对象本身,所以可以将几个参数的调用串成一个链,假设Artisan还有个artisanSex属性,如下

    MapSqlParameterSource mapSqlParameterSource = new MapSqlParameterSource().addValue("artisanName", artisan.getArtisanName()).addValue("artisanSex",artisan.getArtisanSex);

由于这个原因,使用方法调用链模式设计的API很容易使用。


单元测试

@Testpublic void queryTeacherById() {// Artisan artisan = new Artisan();// artisan.setArtisanName("ArtisanNJ");// artisanNJDaoImpl.addArtisan(artisan);Artisan artisan = new Artisan();artisan.setArtisanName("ArtisanMS");artisanNJDaoImpl.addArtisanWithMapSqlParameterSource(artisan);}

数据库结果


NamedParameterJdbcTemplate 支持 in 的操作

PrepareStatement的缺陷

如果我们想查找artisan_id在 1 ,3 , 5 中的数据, PrepareStatement对in的操作只能动态拼接

String in_data = "1,3,5";
pst = conn.prepareStatement("select artisan_name from artisan where artisan_id in (?)");
pst.setString(1,in_data);

使用传统的prepareStatement是动态设定参数的,也就是生成 select artisan_name from artisan where artisan_id in (?) ,一个 ? 代表一个参数,pst.setString(1,”1,3,5”) 就相当于 select artisan_name from artisan where artisan_id in (“1,3,5”) ,这个SQL是查找artisan_id为 “1,3,5”的记录,而不是在 1,3,5中记录。


NamedParameterJdbcTemplate的操作示例

NamedParameterJdbcTemplate可以很好地解决上述问题呢。

....private final static String selectArtisanByIds = "select artisan_name from artisan where artisan_id in (:artisanId)";....public List<Artisan> getArtisanByIds(List<String> artisanIds) {final List<Artisan> artisanList = new ArrayList<Artisan>();// 使用MapSqlParameterSource绑定参数MapSqlParameterSource mapSqlParameterSource = new MapSqlParameterSource();mapSqlParameterSource.addValue("artisanId", artisanIds);namedParameterJdbcTemplate.query(selectArtisanByIds,mapSqlParameterSource, new RowCallbackHandler() {@Overridepublic void processRow(ResultSet rs) throws SQLException {Artisan artisan = new Artisan();artisan.setArtisanName(rs.getString("artisan_name"));// 加入集合artisanList.add(artisan);}});return artisanList;}

单元测试

@Testpublic void queryTeacherById() {List<String> artisanIds = new ArrayList<String>();artisanIds.add("1");artisanIds.add("3");artisanIds.add("5");List<Artisan> artisans = artisanNJDaoImpl.getArtisanByIds(artisanIds);for (Artisan artisan : artisans) {System.out.println("artisanName:" + artisan.getArtisanName());}}

输出:

2017-09-30 02:01:27,648  INFO [main] (AbstractApplicationContext.java:583) - Refreshing org.springframework.context.support.ClassPathXmlApplicationContext@5f0ab5d: startup date [Sat Sep 30 02:01:27 BOT 2017]; root of context hierarchy
2017-09-30 02:01:27,777  INFO [main] (XmlBeanDefinitionReader.java:317) - Loading XML bean definitions from class path resource [com/xgj/dao/namedParameterJdbcTemplate/conf_namedParameterJdbcTemplate.xml]
initContext successfully
artisanName:Xiao2
artisanName:Xiao0
artisanName:Xiao4
2017-09-30 02:01:30,387  INFO [main] (AbstractApplicationContext.java:984) - Closing org.springframework.context.support.ClassPathXmlApplicationContext@5f0ab5d: startup date [Sat Sep 30 02:01:27 BOT 2017]; root of context hierarchy
close context successfully

示例源码

代码已托管到Github—> https://github.com/yangshangwei/SpringMaster

Spring JDBC-NamedParameterJdbcTemplate模板类相关推荐

  1. Spring JDBC事务支持类jdbcTemplate(了解)

    之前的JDBC中,我们通常使用PreparedStatement类实现事务. 下面简单介绍JdbcTemplate的用法: 1.查询单个对象 2.查询集合 3.插入.更新.删除操作 4.批量操作

  2. Spring-Spring MVC + Spring JDBC + Spring Transaction + Maven 构建web登录模块

    概述 功能简介 环境准备 构建工具Maven 数据库脚本Oracle 建立工程 类包及Spring配置文件规划 持久层 建立领域对象 用户领域对象 登录日志领域对象 UserDao LoginLogD ...

  3. Spring JDBC-使用Spring JDBC访问数据库

    概述 使用Spring JDBC 基本的数据操作 更改数据 返回数据库表的自增主键值 批量更改数据 查询数据 使用RowCallbackHandler处理结果集 使用RowMapperT处理结果集 R ...

  4. spring boot demo(spring jdbc访问数据)

    Accessing Relational Data using JDBC with Spring 您将使用Spring JdbcTemplate 构建应用,访问数据库中数据. 下面的简单数据访问逻辑, ...

  5. Spring JDBC 框架一个最简单的Hello World级别的例子

    本地安装mySQL数据库社区版,如果不知道如何安装,可以查看我这篇文章: MySQL社区版的下载和安装 https://blog.csdn.net/i042416/article/details/10 ...

  6. Druid Spring JDBC Servlet 实现登录功能

    Druid 数据库连接池的实现技术 Spring JDBC 需要使用数据库连接池,Spring JDBC 提供非常好用的查询数据.插入数据.更新数据的API 开发步骤: 1.数据库的设计(创建数据库. ...

  7. Spring JDBC和JdbcTemplate CRUD与DataSource示例

    Spring JDBC示例和JdbcTemplate CRUD与DataSource示例 Spring JDBC是本教程的主题.数据库是大多数企业应用程序不可或缺的一部分.因此,当谈到Java EE框 ...

  8. Spring JDBC 示例

    Spring JDBC示例 Spring JDBC是本教程的主题.数据库是大多数企业应用程序不可或缺的一部分.因此,当谈到Java EE框架时,与JDBC的良好集成非常重要. 目录[ 隐藏 ] 1 S ...

  9. 使用Spring JDBC进行数据访问 (JdbcTemplate/NamedParameterJdbcTemplate/SimpleJdbcTemplate/SimpleJdbcCall/Stor)

    http://www.cnblogs.com/webcc/archive/2012/04/11/2442680.html 使用Spring JDBC进行数据访问 11.1. 简介 Spring JDB ...

最新文章

  1. R语言ggplot2可视化在lines线图的尾端添加线图标签、并且去除图例实战
  2. mysql 用一条sql语句修改两个表里的内容,一条sql语句update更新两个表
  3. funny alphabet
  4. 一文详解超参数调优方法
  5. 余弦函数导数推导过程_人工智能数学基础----导数
  6. boost::geometry::densify用法的测试程序
  7. delphi 停电文本数据丢失_概述DCS系统正确停电和上电的步骤
  8. 浏览器上请求URL的全部过程
  9. cuda tensorflow版本对应_Windows10下安装tensorflow-gpu(2.2.0)安装教程(避坑+保姆式教学)...
  10. Photoshop - CMYK 和 RGB 区别是什么?
  11. Python数据分析工具Pandas——数值计算和统计基础
  12. 如何使用音频转换器将多个音频合并为一个音频
  13. 微信小程序Node.js+uniapp学习计划与日程管理系统app
  14. 4_竞赛无人机基本自动飞行支持函数与导航控制函数解析——零基础学习竞赛无人机搭积木式编程
  15. rep论文阅读2:ResRep_Lossless CNN Pruning via Decoupling Remembering and Forgetting
  16. 裴蜀定理的证明与推广应用
  17. CDM计算机辅助,CDM方法学与CDM项目的.pdf
  18. 给排水计算机应用参考文献,给排水专业专著参考文献 给排水专业外文文献怎么找...
  19. python炫酷动画源代码_Python小技巧:如何用Python代码发一个炫酷的朋友圈
  20. 推荐算法的多模型融合

热门文章

  1. Python基础——Anaconda的安装使用
  2. 计算机视觉编程——图像到图像的映射
  3. java web项目的目录结构以及各文件夹的功能是什么eclipse的web目录及各作用
  4. 类属性、类方法;私有属性、公有属性;私有方法、公有方法;静态属性、静态方法;python的命名规范__xx__\__xx\_xx(自用笔记)
  5. python 面向对象(三)多继承
  6. 真人秀制作网站_[BoA] 出道20周年真人秀Nobody Talks To BoA上演与李秀满总制作人充满默契的对话!...
  7. 133. Leetcode 477. 汉明距离总和 (位运算-汉明距离相关题目)
  8. 41. Leetcode 662. 二叉树最大宽度 (二叉树-二叉树性质)
  9. 文巾解题 45. 跳跃游戏 II
  10. 文巾解题 7. 整数反转