[Spring] Spring JDBCTemplate

2024-05-09 05:08
文章标签 java spring jdbctemplate

本文主要是介绍[Spring] Spring JDBCTemplate,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

JdbcTemplate简介

  • 为了使JDBC更加易于使用,Spring在JDBC API定义了一个抽象层,以此建立一个JDBC存取框架
  • 作为Spring JDBC框架的核心,JDBC模板的设计目的是为不同类型的JDBC操作提供模板方法,每个模板方法都能控制整个过程,并允许覆盖过程中的特定任务

JdbcTemplate的使用

  • 加入jar包

    • c3p0-0.9.1.2.jar
    • com.springsource.net.sf.cglib-2.2.0.jar
    • com.springsource.org.aopalliance-1.0.0.jar
    • com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
    • commons-logging-1.1.3.jar
    • mysql-connector-java-5.1.7-bin.jar
    • spring-aop-4.0.0.RELEASE.jar
    • spring-aspects-4.0.0.RELEASE.jar
    • spring-beans-4.0.0.RELEASE.jar
    • spring-context-4.0.0.RELEASE.jar
    • spring-core-4.0.0.RELEASE.jar
    • spring-expression-4.0.0.RELEASE.jar
    • spring-jdbc-4.0.0.RELEASE.jar
    • spring-orm-4.0.0.RELEASE.jar
    • spring-tx-4.0.0.RELEASE.jar
    • spring-web-4.0.0.RELEASE.jar
    • spring-webmvc-4.0.0.RELEASE.jar
  • 建立DB属性文件db.properties(将数据库配置信息放置在Spring配置文件之外)

    jdbc.user=xxxx
    jdbc.password=xxxx
    jdbc.driverClass=com.mysql.jdbc.Driver
    jdbc.jdbcUrl=jdbc:mysql:///xxxx
    jdbc.initPoolSize=x
    jdbc.maxPoolSize=x
  • 创建applicationContext.xml文件

    <?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: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-4.0.xsd"><!--指定数据库配置文件的存放位置--><context:property-placeholder location="classpath:db.properties"/><!--配置C3P0数据源--><bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"><!--EL表达式获取--><property name="user" value="${jdbc.user}"></property><property name="password" value="${jdbc.password}"></property><property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property><property name="driverClass" value="${jdbc.driverClass}"></property><property name="initialPoolSize" value="${jdbc.initPoolSize}"></property><property name="maxPoolSize" value="${jdbc.maxPoolSize}"></property></bean><!-- 配置Spring的JdbcTemplate --><bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"><property name="dataSource" ref="dataSource"></property></bean>
    </beans>
  • 建立实体类

    • Employee.java
    package com.glemontree.spring.jdbc;public class Employee {private Integer id;private String lastName;private String email;private Department department;public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public String getLastName() {return lastName;}public void setLastName(String lastName) {this.lastName = lastName;}public String getEmail() {return email;}public void setEmail(String email) {this.email = email;}public Department getDepartment() {return department;}public void setDepartment(Department department) {this.department = department;}@Overridepublic String toString() {return "Employee [id=" + id + ", lastName=" + lastName + ", email="
    + email + ", department=" + department + "]";
    }

    }

    - Department.java```java
    package com.glemontree.spring.jdbc;public class Department {private Integer id;private String name;public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}@Overridepublic String toString() {return "Department [id=" + id + ", name=" + name + "]";}
    }
  • 建立单元测试类

    package com.glemontree.spring.jdbc;import java.nio.channels.SelectableChannel;
    import java.sql.SQLException;
    import java.util.ArrayList;
    import java.util.List;import javax.sql.DataSource;
    import javax.swing.text.DefaultEditorKit.InsertBreakAction;import org.junit.Test;
    import org.springframework.context.ApplicationContext;
    import org.springframework.context.support.ClassPathXmlApplicationContext;
    import org.springframework.jdbc.core.BeanPropertyRowMapper;
    import org.springframework.jdbc.core.JdbcTemplate;
    import org.springframework.jdbc.core.RowMapper;public class JdbcTest {private ApplicationContext ctx = null;private JdbcTemplate jdbcTemplate;{ctx = new ClassPathXmlApplicationContext("applicationContext.xml");jdbcTemplate = (JdbcTemplate) ctx.getBean("jdbcTemplate");}@Testpublic void test() {DataSource dataSource = ctx.getBean(DataSource.class);try {System.out.println(dataSource.getConnection());} catch (SQLException e) {// TODO Auto-generated catch blocke.printStackTrace();}}@Testpublic void testUpdate() {String sql = "UPDATE employees SET last_name = ? WHERE ID = ?";jdbcTemplate.update(sql, "Jack", 5);}/**
    * 执行批量更新:批量INSERT,UPDAET,DELETE
    * 最后一个参数是Object[]的List类型,因为修改一条记录需要一个Object[],那么多条就需要多个Object[],即List<Object[]>
    */
    @Test
    public void testBatchUpdate() {String sql = "INSERT INTO employees(last_name, email, dept_id) values(?,?,?)";List<Object[]> batchArgs = new ArrayList<Object[]>();batchArgs.add(new Object[]{"AA", "aa@gmail.com", 1});batchArgs.add(new Object[]{"BB", "bb@gmail.com", 1});batchArgs.add(new Object[]{"CC", "cc@gmail.com", 1});batchArgs.add(new Object[]{"DD", "dd@gmail.com", 1});batchArgs.add(new Object[]{"EE", "ee@gmail.com", 1});jdbcTemplate.batchUpdate(sql, batchArgs);
    }/**
    * 从数据库中获取一条记录,实际得到对应的一个对象
    * 注意:不是调用queryForObject(String sql, Class<Employee> requiredType, Object... args)方法
    * 而需要调用queryForObject(String sql, RowMapper<Employee> rowMapper, Object... args)
    * 其中的RowMapper指定如何映射结果集的行,常用的实现类为BeanPropertyRowMapper
    * 使用SQL中列的别名完成列名和类的属性名的映射,例如last_name lastName
    * 不支持级联属性,JdbcTemplate到底是一个JDBC的一个小工具,而不是ORM框架
    */
    @Test
    public void QueryForObject() {String sql = "SELECT id, last_name lastName, email, dept_id as \"department.id\" FROM employees WHERE id = ?";RowMapper<Employee> rowMapper = new BeanPropertyRowMapper<Employee>(Employee.class);Employee employee = jdbcTemplate.queryForObject(sql, rowMapper, 1);System.out.println(employee);
    }/**
    * 查找实体类的集合
    * 注意调用的不是queryForList()方法
    */
    @Test
    public void testQueryForList() {String sql = "SELECT id, last_name lastName, email FROM employees WHERE id > ?";RowMapper<Employee> rowMapper = new BeanPropertyRowMapper<Employee>(Employee.class);List<Employee> employees = jdbcTemplate.query(sql, rowMapper, 5);System.out.println(employees);
    }/*
    * 获取单个列的值,或做统计查询
    * 使用queryForObject(String sql, Class<Long> requiredType)方法
    */
    @Test
    public void testQueryForObject2() {String sql = "SELECT count(id) FROM employees";long count = jdbcTemplate.queryForObject(sql, Long.class);System.out.println(count);
    }
    }

    在单元测试类中进行不同情况的测试,使用JdbcTemplate。

  • 实际使用

    package com.glemontree.spring.jdbc;import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.jdbc.core.BeanPropertyRowMapper;
    import org.springframework.jdbc.core.JdbcTemplate;
    import org.springframework.jdbc.core.RowMapper;
    import org.springframework.stereotype.Repository;@Repository
    public class EmployeeDao {@Autowiredprivate JdbcTemplate jdbcTemplate;public Employee get(Integer id) {String sql = "SELECT id, last_name lastName, email FROM employees WHERE id = ?";RowMapper<Employee> rowMapper = new BeanPropertyRowMapper<Employee>(Employee.class);Employee employee = jdbcTemplate.queryForObject(sql, rowMapper, id);System.out.println(employee);return employee;}
    }   

在JDBC模板中使用具名参数

具名参数简介

  • 在经典的JDBC语法中,SQL参数是用占位符?表示,并且受到位置的限制。定位参数的问题在于一旦参数的顺序发生变化,就必须改变参数绑定
  • 在Spring JDBC框架中,绑定SQL参数的另一种选择是使用具名参数(named parameter)
  • 具名参数:SQL按名称(以冒号开头)而不是按位置进行指定,具名参数更易于维护,也提升了可读性
  • 具名参数只在NamedParameterJdbcTemplate中得到支持

使用方法

  • Spring配置文件中配置NamedParameterJdbcTemplate

    <!-- 配置NamedParameterJdbcTemplate,该对象可以使用具名参数,其没有无参构造器,必须为其构造器指定参数 -->
    <bean id="namedParameterJdbcTemplate" class="org.springframework.jdbc.core.namedparam.NamedParameterJdbcTemplate">
    <constructor-arg ref="dataSource"></constructor-arg>
    </bean>
  • 测试

    @Test
    public void testNamedParameterJdbcTemplate2() {
    String sql = "INSERT INTO employees(last_name, email, dept_id) VALUES(:lastName, :email, :deptId)";
    Employee employee = new Employee();
    employee.setLastName("XYZ");
    employee.setEmail("xyz@gmail.com");
    employee.setDeptId(3);
    SqlParameterSource parameterSource = new BeanPropertySqlParameterSource(employee);
    namedParameterJdbcTemplate.update(sql, parameterSource);
    }

    注意具名参数使用:参数名来表示。

    使用具名参数时可以使用update(String sql, SqlParameterSource paramSource)这个方法来进行更新操作:

    • SQL语句中的参数名和类的属性名一致
    • 使用SqlParameterSource的BeanPropertySqlParameterSource实现类作为参数

这篇关于[Spring] Spring JDBCTemplate的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

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. 内存

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参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空

深度解析Java DTO(最新推荐)

《深度解析JavaDTO(最新推荐)》DTO(DataTransferObject)是一种用于在不同层(如Controller层、Service层)之间传输数据的对象设计模式,其核心目的是封装数据,... 目录一、什么是DTO?DTO的核心特点:二、为什么需要DTO?(对比Entity)三、实际应用场景解析

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

深度解析Java项目中包和包之间的联系

《深度解析Java项目中包和包之间的联系》文章浏览阅读850次,点赞13次,收藏8次。本文详细介绍了Java分层架构中的几个关键包:DTO、Controller、Service和Mapper。_jav... 目录前言一、各大包1.DTO1.1、DTO的核心用途1.2. DTO与实体类(Entity)的区别1