日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

SpringBoot2.x入門教程之引入jdbc模塊與JdbcTemplate簡單使用方法

瀏覽:21日期:2023-05-05 13:42:32

前提

這篇文章是《SpringBoot2.x入門》專輯的第7篇文章,使用的SpringBoot版本為2.3.1.RELEASE,JDK版本為1.8。

這篇文章會簡單介紹jdbc模塊也就是spring-boot-starter-jdbc組件的引入、數據源的配置以及JdbcTemplate的簡單使用。為了讓文中的例子相對通用,下文選用MySQL8.x、h2database(內存數據庫)作為示例數據庫,選用主流的Druid和HikariCP作為示例數據源。

引入jdbc模塊

引入spring-boot-starter-jdbc組件,如果在父POM全局管理spring-boot依賴版本的前提下,只需要在項目pom文件的dependencies元素直接引入:

<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jdbc</artifactId></dependency>

通過IDEA展開該依賴的關系圖如下:

SpringBoot2.x入門教程之引入jdbc模塊與JdbcTemplate簡單使用方法

其實spring-boot-starter-jdbc模塊本身已經引入了spring-jdbc(間接引入spring-core、spring-beans、spring-tx)、spring-boot-starter和HikariCP三個依賴,如果希望啟動Servlet容器,可以額外引入spring-boot-starter-jdbc。

spring-boot-starter-jdbc提供了數據源配置、事務管理、數據訪問等等功能,而對于不同類型的數據庫,需要提供不同的驅動實現,才能更加簡單地通過驅動實現根據連接URL、用戶口令等屬性直接連接數據庫(或者說獲取數據庫的連接),因此對于不同類型的數據庫,需要引入不同的驅動包依賴。對于MySQL而言,需要引入mysql-connector-java,而對于h2database而言,需要引入h2(驅動包和數據庫代碼位于同一個依賴中),兩者中都具備數據庫抽象驅動接口java.sql.Driver的實現類:

對于mysql-connector-java而言,常用的實現是com.mysql.cj.jdbc.Driver(MySQL8.x版本)。 對于h2而言,常用的實現是org.h2.Driver。

如果需要連接的數據庫是h2database,引入h2對應的數據庫和驅動依賴如下:

<dependency> <groupId>com.h2database</groupId> <artifactId>h2</artifactId> <version>1.4.200</version></dependency>

如果需要連接的數據庫是MySQL,引入MySQL對應的驅動依賴如下:

<dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>8.0.20</version></dependency>

上面的類庫版本選取了編寫本文時候的最新版本,實際上要根據軟件對應的版本選擇合適的驅動版本。

數據源配置

spring-boot-starter-jdbc模塊默認使用HikariCP作為數據庫的連接池。

HikariCP,也就是Hikari Connection Pool,Hikari連接池。HikariCP的作者是日本人,而Hikari是日語,意義和light相近,也就是'光'。Simplicity is prerequisite for reliability(簡單是可靠的先決條件)是HikariCP的設計理念,他是一款代碼精悍的高性能連接池框架,被Spring項目選中作為內建默認連接池,值得信賴。

如果決定使用HikariCP連接h2數據庫,則配置文件中添加如下的配置項以配置數據源HikariDataSource:

spring.datasource.driver-class-name=org.h2.Driverspring.datasource.url=jdbc:h2:mem:testspring.datasource.username=rootspring.datasource.password=123456# 可選配置,是否啟用h2數據庫的WebUI控制臺spring.h2.console.enabled=true# 可選配置,訪問h2數據庫的WebUI控制臺的路徑spring.h2.console.path=/h2-console# 可選配置,是否允許非本機訪問h2數據庫的WebUI控制臺spring.h2.console.settings.web-allow-others=true

SpringBoot2.x入門教程之引入jdbc模塊與JdbcTemplate簡單使用方法

如果決定使用HikariCP連接MySQL數據庫,則配置文件中添加如下的配置項以配置數據源HikariDataSource:

spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver# 注意MySQL8.x需要指定服務時區屬性spring.datasource.url=jdbc:mysql://localhost:3306/local?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=UTF-8&zeroDateTimeBehavior=convertToNull&useSSL=falsespring.datasource.username=rootspring.datasource.password=root

有時候可能更偏好于使用其他連接池,例如Alibaba出品的Durid,這樣就要禁用默認的數據源加載,改成Durid提供的數據源。引入Druid數據源需要額外添加依賴:

<dependency> <groupId>com.alibaba</groupId> <artifactId>druid</artifactId> <version>1.1.23</version></dependency>

如果決定使用Druid連接MySQL數據庫,則配置文件中添加如下的配置項以配置數據源DruidDataSource:

spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver# 注意MySQL8.x需要指定服務時區屬性spring.datasource.url=jdbc:mysql://localhost:3306/local?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=UTF-8&zeroDateTimeBehavior=convertToNull&useSSL=falsespring.datasource.username=rootspring.datasource.password=root# 指定數據源類型為Druid提供的數據源spring.datasource.type=com.alibaba.druid.pool.DruidDataSource

上面這樣配置DruidDataSource,所有數據源的屬性值都會選用默認值,如果想深度定制數據源的屬性,則需要覆蓋由DataSourceConfiguration.Generic創建的數據源,先預設所有需要的配置,為了和內建的spring.datasource屬性前綴避嫌,這里自定義一個屬性前綴druid,配置文件中添加自定義配置項如下:

druid.url=jdbc:mysql://localhost:3306/local?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=UTF-8&zeroDateTimeBehavior=convertToNull&useSSL=falsedruid.driver-class-name=com.mysql.cj.jdbc.Driverdruid.username=rootdruid.password=root# 初始化大小druid.initialSize=1# 最大druid.maxActive=20# 空閑druid.minIdle=5# 配置獲取連接等待超時的時間druid.maxWait=60000# 配置間隔多久才進行一次檢測,檢測需要關閉的空閑連接,單位是毫秒druid.timeBetweenEvictionRunsMillis=60000# 配置一個連接在池中最小生存的時間,單位是毫秒druid.minEvictableIdleTimeMillis=60000druid.validationQuery=SELECT 1 FROM DUALdruid.testWhileIdle=truedruid.testOnBorrow=falsedruid.testOnReturn=false# 打開PSCache,并且指定每個連接上PSCache的大小druid.poolPreparedStatements=truedruid.maxPoolPreparedStatementPerConnectionSize=20# 配置監控統計攔截的filters,后臺統計相關druid.filters=stat,wall# 打開mergeSql功能;慢SQL記錄druid.connectionProperties=druid.stat.mergeSql=true;druid.stat.slowSqlMillis=5000

這里要確保本地安裝了一個8.x版本的MySQL服務,并且建立了一個命名為local的數據庫。

需要在項目中添加一個數據源自動配置類,這里命名為DruidAutoConfiguration,通過注解@ConfigurationProperties把druid前綴的屬性注入到數據源實例中:

@Configurationpublic class DruidAutoConfiguration { @Bean @ConfigurationProperties(prefix = 'druid') public DataSource dataSource() { return new DruidDataSource(); } @Bean public ServletRegistrationBean<StatViewServlet> statViewServlet() { ServletRegistrationBean<StatViewServlet> servletRegistrationBean = new ServletRegistrationBean<>(new StatViewServlet(), '/druid/*'); // 添加IP白名單 servletRegistrationBean.addInitParameter('allow', '127.0.0.1'); // 添加控制臺管理用戶 servletRegistrationBean.addInitParameter('loginUsername', 'admin'); servletRegistrationBean.addInitParameter('loginPassword', '123456'); // 是否能夠重置數據 servletRegistrationBean.addInitParameter('resetEnable', 'true'); return servletRegistrationBean; } @Bean public FilterRegistrationBean<WebStatFilter> webStatFilter() { WebStatFilter webStatFilter = new WebStatFilter(); FilterRegistrationBean<WebStatFilter> filterRegistrationBean = new FilterRegistrationBean<>(); filterRegistrationBean.setFilter(webStatFilter); // 添加過濾規則 filterRegistrationBean.addUrlPatterns('/*'); // 忽略過濾格式 filterRegistrationBean.addInitParameter('exclusions', '*.js,*.gif,*.jpg,*.png,*.css,*.ico,/druid/*,'); return filterRegistrationBean; }}

可以通過訪問${requestContext}/druid/login.html跳轉到Druid的監控控制臺,登錄賬號密碼就是在statViewServlet中配置的用戶和密碼:

SpringBoot2.x入門教程之引入jdbc模塊與JdbcTemplate簡單使用方法

Druid是一款爭議比較多的數據源框架,項目的Issue中也有人提出過框架中加入太多和連接池無關的功能,例如SQL監控、屬性展示等等,這些功能本該讓專業的監控軟件完成。但毫無疑問,這是一款活躍度比較高的優秀國產開源框架。

配置schema和data腳本

spring-boot-starter-jdbc可以通過一些配置然后委托DataSourceInitializerInvoker進行schema(一般理解為DDL)和data(一般理解為DML)腳本的加載和執行,具體的配置項是:

# 定義schema的加載路徑,可以通過英文逗號指定多個路徑spring.datasource.schema=classpath:/ddl/schema.sql# 定義data的加載路徑,可以通過英文逗號指定多個路徑spring.datasource.data=classpath:/dml/data.sql# 可選# spring.datasource.schema-username=# spring.datasource.schema-password=# 項目數據源初始化之后的執行模式,可選值EMBEDDED、ALWAYS和NEVERspring.datasource.initialization-mode=always

類路徑的resources文件夾下添加ddl/schema.sql:

DROP TABLE IF EXISTS customer;CREATE TABLE customer( id BIGINT UNSIGNED AUTO_INCREMENT PRIMARY KEY COMMENT ’主鍵’, customer_name VARCHAR(32) NOT NULL COMMENT ’客戶名稱’, create_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT ’創建時間’, edit_time DATETIME NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT ’修改時間’) COMMENT ’客戶表’;

由于spring.datasource.initialization-mode指定為ALWAYS,每次數據源初始化都會執行spring.datasource.schema中配置的腳本,會刪表重建。接著類路徑的resources文件夾下添加dml/data.sql:

INSERT INTO customer(customer_name) VALUES (’throwable’);

添加一個CommandLineRunner實現驗證一下:

@Slf4j@SpringBootApplicationpublic class Ch7Application implements CommandLineRunner { @Autowired private DataSource dataSource; public static void main(String[] args) { SpringApplication.run(Ch7Application.class, args); } @Override public void run(String... args) throws Exception { Connection connection = dataSource.getConnection(); ResultSet resultSet = connection.createStatement().executeQuery('SELECT * FROM customer WHERE id = 1'); while (resultSet.next()) { log.info('id:{},name:{}', resultSet.getLong('id'), resultSet.getString('customer_name')); } resultSet.close(); connection.close(); }}

啟動后執行結果如下:

SpringBoot2.x入門教程之引入jdbc模塊與JdbcTemplate簡單使用方法

這里務必注意一點,spring.datasource.schema指定的腳本執行成功之后才會執行spring.datasource.data指定的腳本,如果想僅僅執行spring.datasource.data指定的腳本,那么需要至少把spring.datasource.schema指向一個空的文件,確保spring.datasource.schema指定路徑的文件初始化成功。

使用JdbcTemplate

spring-boot-starter-jdbc中自帶的JdbcTemplate是對JDBC的輕度封裝。這里只簡單介紹一下它的使用方式,構建一個面向前面提到的customer表的具備CURD功能的DAO。這里先在前文提到的DruidAutoConfiguration中添加一個JdbcTemplate實例到IOC容器中:

@Beanpublic JdbcTemplate jdbcTemplate(DataSource dataSource){ return new JdbcTemplate(dataSource);}

添加一個Customer實體類:

// 實體類@Datapublic class Customer { private Long id; private String customerName; private LocalDateTime createTime; private LocalDateTime editTime;}

接著添加一個CustoemrDao類,實現增刪改查:

// CustoemrDao@RequiredArgsConstructor@Repositorypublic class CustomerDao { private final JdbcTemplate jdbcTemplate; /** * 增 */ public int insertSelective(Customer customer) { StringJoiner p = new StringJoiner(',', '(', ')'); StringJoiner v = new StringJoiner(',', '(', ')'); Optional.ofNullable(customer.getCustomerName()).ifPresent(x -> { p.add('customer_name'); v.add('?'); }); Optional.ofNullable(customer.getCreateTime()).ifPresent(x -> { p.add('create_time'); v.add('?'); }); Optional.ofNullable(customer.getEditTime()).ifPresent(x -> { p.add('edit_time'); v.add('?'); }); String sql = 'INSERT INTO customer' + p.toString() + ' VALUES ' + v.toString(); KeyHolder keyHolder = new GeneratedKeyHolder(); int updateCount = jdbcTemplate.update(con -> { PreparedStatement ps = con.prepareStatement(sql, Statement.RETURN_GENERATED_KEYS); int index = 1; if (null != customer.getCustomerName()) { ps.setString(index++, customer.getCustomerName()); } if (null != customer.getCreateTime()) { ps.setTimestamp(index++, Timestamp.valueOf(customer.getCreateTime())); } if (null != customer.getEditTime()) { ps.setTimestamp(index, Timestamp.valueOf(customer.getEditTime())); } return ps; }, keyHolder); customer.setId(Objects.requireNonNull(keyHolder.getKey()).longValue()); return updateCount; } /** * 刪 */ public int delete(long id) { return jdbcTemplate.update('DELETE FROM customer WHERE id = ?', id); } /** * 查 */ public Customer queryByCustomerName(String customerName) { return jdbcTemplate.query('SELECT * FROM customer WHERE customer_name = ?', ps -> ps.setString(1, customerName), SINGLE); } public List<Customer> queryAll() { return jdbcTemplate.query('SELECT * FROM customer', MULTI); } public int updateByPrimaryKeySelective(Customer customer) { final long id = Objects.requireNonNull(Objects.requireNonNull(customer).getId()); StringBuilder sql = new StringBuilder('UPDATE customer SET '); Optional.ofNullable(customer.getCustomerName()).ifPresent(x -> sql.append('customer_name = ?,')); Optional.ofNullable(customer.getCreateTime()).ifPresent(x -> sql.append('create_time = ?,')); Optional.ofNullable(customer.getEditTime()).ifPresent(x -> sql.append('edit_time = ?,')); StringBuilder q = new StringBuilder(sql.substring(0, sql.lastIndexOf(','))).append(' WHERE id = ?'); return jdbcTemplate.update(q.toString(), ps -> { int index = 1; if (null != customer.getCustomerName()) { ps.setString(index++, customer.getCustomerName()); } if (null != customer.getCreateTime()) { ps.setTimestamp(index++, Timestamp.valueOf(customer.getCreateTime())); } if (null != customer.getEditTime()) { ps.setTimestamp(index++, Timestamp.valueOf(customer.getEditTime())); } ps.setLong(index, id); }); } private static Customer convert(ResultSet rs) throws SQLException { Customer customer = new Customer(); customer.setId(rs.getLong('id')); customer.setCustomerName(rs.getString('customer_name')); customer.setCreateTime(rs.getTimestamp('create_time').toLocalDateTime()); customer.setEditTime(rs.getTimestamp('edit_time').toLocalDateTime()); return customer; } private static ResultSetExtractor<List<Customer>> MULTI = rs -> { List<Customer> result = new ArrayList<>(); while (rs.next()) { result.add(convert(rs)); } return result; }; private static ResultSetExtractor<Customer> SINGLE = rs -> rs.next() ? convert(rs) : null;}

測試結果如下:

SpringBoot2.x入門教程之引入jdbc模塊與JdbcTemplate簡單使用方法

JdbcTemplate的優勢是可以應用函數式接口簡化一些值設置和值提取的操作,并且獲得接近于原生JDBC的執行效率,但是它的明顯劣勢就是會產生大量模板化的代碼,在一定程度上影響開發效率。

小結

本文簡單分析spring-boot-starter-jdbc引入,以及不同數據庫和不同數據源的使用方式,最后簡單介紹了JdbcTemplate的基本使用。

demo項目倉庫:

Github:https://github.com/zjcscut/spring-boot-guide/tree/master/ch6-jdbc-module-h2

Github:https://github.com/zjcscut/spring-boot-guide/tree/master/ch7-jdbc-module-mysql

總結

到此這篇關于SpringBoot2.x入門教程:引入jdbc模塊與JdbcTemplate簡單使用的文章就介紹到這了,更多相關SpringBoot2.x入門教程:引入jdbc模塊與JdbcTemplate簡單使用內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Spring
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产成人免费视频网站视频社区| 国产一区2区| 成人久久一区| 国产在线观看www| av在线最新| 久久久精品午夜少妇| 欧美一级精品| 国产麻豆精品久久| 久久av导航| 久久久久九九精品影院| 国产激情欧美| а√在线中文在线新版| 久久久蜜桃一区二区人| 狠狠久久婷婷| 美国欧美日韩国产在线播放| 在线国产日韩| 欧美日韩18| 精品日韩一区| 五月激情久久| 视频在线观看国产精品| 日本在线成人| 精品亚洲精品| 群体交乱之放荡娇妻一区二区| 亚洲成av人片一区二区密柚| 久久国产亚洲| 首页欧美精品中文字幕| 日韩av午夜在线观看| 欧美xxxx性| 久久九九电影| 六月天综合网| 欧美久久一区二区三区| 国产一区二区三区久久| 亚洲午夜电影| 亚洲精一区二区三区| 欧美国产免费| 亚洲性色视频| 91免费精品国偷自产在线在线| 麻豆精品少妇| 风间由美中文字幕在线看视频国产欧美| 精品三级国产| 亚洲激情国产| 国产精品久久久久毛片大屁完整版| 久久久久久夜| 久久亚洲美女| 日韩免费高清| 中文字幕免费一区二区| 麻豆久久久久久| 婷婷久久一区| 国产精品视频一区二区三区四蜜臂| 中文一区一区三区高中清不卡免费| 欧美日韩少妇| 国产精品国产三级在线观看| 999久久久91| 日本不卡不码高清免费观看| 精品国产欧美日韩| 亚洲一区网站| 精品国产亚洲日本| 久久av在线| 97精品国产| 日韩精品免费观看视频| 亚洲伦乱视频| 国产欧美在线| 国产亚洲高清视频| 精品中国亚洲| 中文不卡在线| 日韩三区免费| 国产欧美日韩精品高清二区综合区 | 免费在线观看成人| 免费亚洲婷婷| 蜜桃91丨九色丨蝌蚪91桃色| 欧美三级网址| 国产精品白丝久久av网站| 在线亚洲激情| 伊人久久视频| 国产美女久久| 综合国产精品| 999久久久91| 国产一区二区三区不卡av| 午夜一区在线| 国产99久久| 欧美激情另类| 国产伦精品一区二区三区千人斩| 激情婷婷综合| 激情黄产视频在线免费观看| 国产亚洲一区二区三区不卡| 丝袜国产日韩另类美女| 欧美sss在线视频| 国产一区二区三区不卡av| 国产亚洲字幕| 日韩精品午夜视频| 蜜臀av在线播放一区二区三区| 韩国精品主播一区二区在线观看| 国产精品chinese| 视频一区国产视频| 久久国产中文字幕| 色偷偷偷在线视频播放| 欧美国产极品| 国产欧美亚洲一区| 婷婷亚洲精品| 亚洲天堂日韩在线| 99国产精品久久久久久久| 九九精品调教| 国产传媒在线观看| 精品一区电影| 国产高清亚洲| 国产精品久久久久久久久久妞妞| 亚洲毛片一区| 亚洲永久精品唐人导航网址| 午夜欧美精品| 国产私拍福利精品视频二区| 深夜福利一区| 亚洲精品影视| 亚洲va久久久噜噜噜久久| 综合亚洲色图| 亚洲精品进入| 日韩一区免费| 亚洲精品亚洲人成在线观看| 在线精品国产亚洲| 亚洲综合专区| 亚洲精品视频一二三区| 亚洲精品进入| 亚洲精品日韩久久| 日韩一区二区三区高清在线观看| 蜜臀国产一区二区三区在线播放| 久久国产精品毛片| 日韩精品一二三四| 中文一区一区三区免费在线观| 一二三区精品| 色婷婷成人网| 国产日韩视频在线| 免费精品一区| 97精品国产福利一区二区三区| 欧美日韩在线观看首页| 电影亚洲精品噜噜在线观看| 久久精品导航| 亚洲一区二区免费看| 中文字幕视频精品一区二区三区| 日韩欧美高清一区二区三区| 日韩**一区毛片| 国产欧美91| 欧美激情另类| 亚洲二区视频| 免费一级片91| 日韩成人精品一区二区三区| 国产麻豆精品| av资源亚洲| 狠狠色狠狠色综合日日tαg| 亚洲精选成人| 久久99免费视频| 日韩三区在线| 鲁大师成人一区二区三区| 欧美日韩一区二区三区四区在线观看| 免费一级欧美在线观看视频| 香蕉成人av| 一区在线视频观看| 亚州国产精品| 久久精品网址| 亚洲成人一区在线观看| 欧美日韩国产在线一区| 婷婷综合一区| 国产成人精品一区二区三区视频| 久久久一二三| 蜜臀久久99精品久久久久久9| 国产欧美日韩精品高清二区综合区| 国产一区二区三区不卡av| 五月婷婷六月综合| 91成人在线精品视频| 中文在线中文资源| 91九色精品| 国产伦精品一区二区三区千人斩| 三级精品视频| 欧美亚洲一级| 韩国精品主播一区二区在线观看| 中文字幕日韩欧美精品高清在线| 久久影院资源站| 国产农村妇女精品一二区 | 亚洲最大av| 97精品一区| 亚洲免费毛片| 久久蜜桃精品| 国产精品乱战久久久| 成人av二区| 欧美一区二区三区久久精品| 日韩欧美三级| 日本不卡一区二区三区| 欧美日韩免费看片| 日韩欧美精品一区二区综合视频| 麻豆理论在线观看| 丝袜a∨在线一区二区三区不卡| 久久精品日韩欧美| 石原莉奈在线亚洲三区| 手机在线电影一区| 日韩精品午夜视频| av一区在线| 国产视频网站一区二区三区| 亚洲成人va| 国产福利资源一区| 香蕉久久国产| 黄色在线网站噜噜噜| 欧美在线不卡|