天天看點

Spring JDBC-NamedParameterJdbcTemplate模闆類概述示例NamedParameterJdbcTemplate 支援 in 的操作示例源碼

  • 概述
  • 示例
    • 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
     */
    @Autowired
    public void setNamedParameterJdbcTemplate(
            NamedParameterJdbcTemplate namedParameterJdbcTemplate) {
        this.namedParameterJdbcTemplate = namedParameterJdbcTemplate;
    }

    @Override
    public void addArtisan(Artisan artisan) {

        // 定義命名參數
        SqlParameterSource sps = new BeanPropertySqlParameterSource(artisan);
        // 使用模闆類方法
        namedParameterJdbcTemplate.update(insertArtisanSql, sps);
    }

}           

複制

在SQL語句中聲明命名參數的格式為

:paranName           

複制

比如values(:artisanName) ,多個參數使用逗号分隔。

在這個示例中,使用BeanPropertySqlParameterSource提供資料源,它接收一個JavaBean作為構造函數的入參,調用namedParameterJdbcTemplate.update(insertArtisanSql, sps)執行插入資料的而操作。

配置檔案

<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.xsd
       http://www.springframework.org/schema/context 
       http://www.springframework.org/schema/context/spring-context.xsd">

    
    <context:component-scan base-package="com.xgj.dao.namedParameterJdbcTemplate" />


    
    <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}" />


    
    <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;

    @Before
    public 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");
    }

    @Test
    public void queryTeacherById() {
        Artisan artisan = new Artisan();
        artisan.setArtisanName("ArtisanNJ");
        artisanNJDaoImpl.addArtisan(artisan);
    }

    @After
    public void closeContext() {
        if (ctx != null) {
            ctx.close();
        }
        System.out.println("close context successfully");
    }
}           

複制

結果:

檢視下資料庫是否插入成功 (隻是示範,忽略ID…)

Spring JDBC-NamedParameterJdbcTemplate模闆類概述示例NamedParameterJdbcTemplate 支援 in 的操作示例源碼

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很容易使用。

單元測試

@Test
    public void queryTeacherById() {
        // Artisan artisan = new Artisan();
        // artisan.setArtisanName("ArtisanNJ");
        // artisanNJDaoImpl.addArtisan(artisan);

        Artisan artisan = new Artisan();
        artisan.setArtisanName("ArtisanMS");
        artisanNJDaoImpl.addArtisanWithMapSqlParameterSource(artisan);
    }           

複制

資料庫結果

Spring JDBC-NamedParameterJdbcTemplate模闆類概述示例NamedParameterJdbcTemplate 支援 in 的操作示例源碼

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 getArtisanByIds(List artisanIds) {

        final List artisanList = new ArrayList();

        // 使用MapSqlParameterSource綁定參數
        MapSqlParameterSource mapSqlParameterSource = new MapSqlParameterSource();

        mapSqlParameterSource.addValue("artisanId", artisanIds);

        namedParameterJdbcTemplate.query(selectArtisanByIds,
                mapSqlParameterSource, new RowCallbackHandler() {

                    @Override
                    public void processRow(ResultSet rs) throws SQLException {
                        Artisan artisan = new Artisan();
                        artisan.setArtisanName(rs.getString("artisan_name"));
                        // 加入集合
                        artisanList.add(artisan);
                    }
                });

        return artisanList;
    }           

複制

單元測試

@Test
    public void queryTeacherById() {
        List artisanIds = new ArrayList();
        artisanIds.add("1");
        artisanIds.add("3");
        artisanIds.add("5");
        List 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