Spring Boot 集成 Sharding-jdbc + Mybatis-Plus 實(shí)現(xiàn)分庫分表

一诈铛、 Sharding-jdbc簡介

Sharding-jdbc是開源的數(shù)據(jù)庫操作中間件勿璃;定位為輕量級Java框架颊埃,在Java的JDBC層提供的額外服務(wù)仁讨。它使用客戶端直連數(shù)據(jù)庫,以jar包形式提供服務(wù),無需額外部署和依賴猬错,可理解為增強(qiáng)版的JDBC驅(qū)動窗看,完全兼容JDBC和各種ORM框架。

官方文檔地址:https://shardingsphere.apache.org/document/current/cn/overview/

本文demo實(shí)現(xiàn)了分庫分表功能倦炒。

二显沈、項(xiàng)目結(jié)構(gòu)

首先創(chuàng)建一個一般的Spring boot項(xiàng)目,項(xiàng)目采用三層架構(gòu)析校,結(jié)構(gòu)圖如下:項(xiàng)目目錄結(jié)構(gòu)圖 pom.xml文件如下:

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

? ? ? ? xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

? ? <modelVersion>4.0.0</modelVersion>

? ? <parent>

? ? ? ? <groupId>org.springframework.boot</groupId>

? ? ? ? <artifactId>spring-boot-starter-parent</artifactId>

? ? ? ? <version>2.1.6.RELEASE</version>

? ? ? ? <relativePath/> <!-- lookup parent from repository -->

? ? </parent>

? ? <groupId>com.macky</groupId>

? ? <artifactId>spring-boot-shardingjdbc</artifactId>

? ? <version>0.0.1-SNAPSHOT</version>

? ? <name>spring-boot-shardingjdbc</name>

? ? <description>Demo project for spring-boot-shardingjdbc</description>

? ? <properties>

? ? ? ? <java.version>1.8</java.version>

? ? </properties>

? ? <dependencies>

? ? ? ? <dependency>

? ? ? ? ? ? <groupId>org.springframework.boot</groupId>

? ? ? ? ? ? <artifactId>spring-boot-starter-web</artifactId>

? ? ? ? </dependency>

? ? ? ? <dependency>

? ? ? ? ? ? <groupId>org.springframework.boot</groupId>

? ? ? ? ? ? <artifactId>spring-boot-starter-test</artifactId>

? ? ? ? ? ? <scope>test</scope>

? ? ? ? </dependency>

? ? ? ? <!--mysql-->

? ? ? ? <dependency>

? ? ? ? ? ? <groupId>mysql</groupId>

? ? ? ? ? ? <artifactId>mysql-connector-java</artifactId>

? ? ? ? ? ? <scope>runtime</scope>

? ? ? ? </dependency>

? ? ? ? <!--Mybatis-Plus-->

? ? ? ? <dependency>

? ? ? ? ? ? <groupId>com.baomidou</groupId>

? ? ? ? ? ? <artifactId>mybatis-plus-boot-starter</artifactId>

? ? ? ? ? ? <version>3.1.1</version>

? ? ? ? </dependency>

? ? ? ? <!--shardingsphere start-->

? ? ? ? <!-- for spring boot -->

? ? ? ? <dependency>

? ? ? ? ? ? <groupId>io.shardingsphere</groupId>

? ? ? ? ? ? <artifactId>sharding-jdbc-spring-boot-starter</artifactId>

? ? ? ? ? ? <version>3.1.0</version>

? ? ? ? </dependency>

? ? ? ? <!-- for spring namespace -->

? ? ? ? <dependency>

? ? ? ? ? ? <groupId>io.shardingsphere</groupId>

? ? ? ? ? ? <artifactId>sharding-jdbc-spring-namespace</artifactId>

? ? ? ? ? ? <version>3.1.0</version>

? ? ? ? </dependency>

? ? ? ? <!--shardingsphere end-->

? ? ? ? <!--lombok-->

? ? ? ? <dependency>

? ? ? ? ? ? <groupId>org.projectlombok</groupId>

? ? ? ? ? ? <artifactId>lombok</artifactId>

? ? ? ? </dependency>

? ? </dependencies>

? ? <build>

? ? ? ? <plugins>

? ? ? ? ? ? <plugin>

? ? ? ? ? ? ? ? <groupId>org.springframework.boot</groupId>

? ? ? ? ? ? ? ? <artifactId>spring-boot-maven-plugin</artifactId>

? ? ? ? ? ? </plugin>

? ? ? ? </plugins>

? ? </build>

</project>

實(shí)體類以書本為例:

package com.macky.springbootshardingjdbc.entity;

import com.baomidou.mybatisplus.annotation.TableName;

import com.baomidou.mybatisplus.extension.activerecord.Model;

import groovy.transform.EqualsAndHashCode;

import lombok.Data;

import lombok.experimental.Accessors;

/**

* @author Macky

* @Title class Book

* @Description: 書籍是實(shí)體類

* @date 2019/7/13 15:23

*/

@Data

@EqualsAndHashCode(callSuper = true)

@Accessors(chain = true)

@TableName("book")

public class Book extends Model<Book> {

? ? private int id;

? ? private String name;

? ? private int count;

}

開放保存和查詢兩個接口,代碼如下:

package com.macky.springbootshardingjdbc.controller;

import com.macky.springbootshardingjdbc.entity.Book;

import com.macky.springbootshardingjdbc.service.BookService;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.web.bind.annotation.*;

import java.util.List;

/**

* @author Macky

* @Title class BookController

* @Description: TODO

* @date 2019/7/12 20:53

*/

@RestController

public class BookController {

? ? @Autowired

? ? BookService bookService;

? ? @RequestMapping(value = "/book", method = RequestMethod.GET)

? ? public List<Book> getItems(){

? ? ? ? return bookService.getBookList();

? ? }

? ? @RequestMapping(value = "/book",method = RequestMethod.POST)

? ? public Boolean saveItem(Book book){

? ? ? ? return bookService.save(book);

? ? }

}

BookServiceImpl.java

package com.macky.springbootshardingjdbc.service.impl;

import com.baomidou.mybatisplus.core.toolkit.Wrappers;

import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;

import com.macky.springbootshardingjdbc.entity.Book;

import com.macky.springbootshardingjdbc.mapper.BookMapper;

import com.macky.springbootshardingjdbc.service.BookService;

import org.springframework.stereotype.Service;

import java.util.List;

/**

* @author Macky

* @Title class BookServiceImpl

* @Description: TODO

* @date 2019/7/12 20:47

*/

@Service

public class BookServiceImpl extends ServiceImpl<BookMapper, Book> implements BookService {

? ? @Override

? ? public List<Book> getBookList() {

? ? ? ? return baseMapper.selectList(Wrappers.<Book>lambdaQuery());

? ? }

? ? @Override

? ? public boolean save(Book book) {

? ? ? ? return super.save(book);

? ? }

}

BookMapper.java

package com.macky.springbootshardingjdbc.mapper;

import com.baomidou.mybatisplus.core.mapper.BaseMapper;

import com.macky.springbootshardingjdbc.entity.Book;

/**

* @author Macky

* @Title class BookMapper

* @Description: TODO

* @date 2019/7/12 20:46

*/

public interface BookMapper extends BaseMapper<Book> {

}

創(chuàng)建數(shù)據(jù)庫表铜涉,DDL語句如下:

#創(chuàng)建數(shù)據(jù)庫表數(shù)據(jù)

CREATE DATABASE IF NOT EXISTS `db0`;

USE `db0`;

DROP TABLE IF EXISTS `book_0`;

CREATE TABLE `book_0` (

? ? `id` INT ( 11 ) NOT NULL,

? ? `name` VARCHAR ( 255 ) DEFAULT NULL,

? ? `count` INT ( 11 ) DEFAULT NULL,

? ? PRIMARY KEY ( `id` )

) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;

DROP TABLE IF EXISTS `book_1`;

CREATE TABLE `book_1` (

? ? `id` INT ( 11 ) NOT NULL,

? ? `name` VARCHAR ( 255 ) DEFAULT NULL,

? ? `count` INT ( 11 ) DEFAULT NULL,

? ? PRIMARY KEY ( `id` )

) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;

CREATE DATABASE IF NOT EXISTS `db1`;

USE `db1`;

DROP TABLE IF EXISTS `book_0`;

CREATE TABLE `book_0` (

? ? `id` INT ( 11 ) NOT NULL,

? ? `name` VARCHAR ( 255 ) DEFAULT NULL,

? ? `count` INT ( 11 ) DEFAULT NULL,

? ? PRIMARY KEY ( `id` )

) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;

DROP TABLE IF EXISTS `book_1`;

CREATE TABLE `book_1` (

? ? `id` INT ( 11 ) NOT NULL,

? ? `name` VARCHAR ( 255 ) DEFAULT NULL,

? ? `count` INT ( 11 ) DEFAULT NULL,

? ? PRIMARY KEY ( `id` )

) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;

CREATE DATABASE IF NOT EXISTS `db2`;

USE `db2`;

DROP TABLE IF EXISTS `book_0`;

CREATE TABLE `book_0` (

? ? `id` INT ( 11 ) NOT NULL,

? ? `name` VARCHAR ( 255 ) DEFAULT NULL,

? ? `count` INT ( 11 ) DEFAULT NULL,

? ? PRIMARY KEY ( `id` )

) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;

DROP TABLE IF EXISTS `book_1`;

CREATE TABLE `book_1` (

? ? `id` INT ( 11 ) NOT NULL,

? ? `name` VARCHAR ( 255 ) DEFAULT NULL,

? ? `count` INT ( 11 ) DEFAULT NULL,

? ? PRIMARY KEY ( `id` )

) ENGINE = INNODB DEFAULT CHARSET = utf8mb4;

配置分庫分表策略application.properties:

# 數(shù)據(jù)源 db0,db1,db2

sharding.jdbc.datasource.names=db0,db1,db2

# 第一個數(shù)據(jù)庫

sharding.jdbc.datasource.db0.type=com.zaxxer.hikari.HikariDataSource

sharding.jdbc.datasource.db0.driver-class-name=com.mysql.cj.jdbc.Driver

sharding.jdbc.datasource.db0.jdbc-url=jdbc:mysql://localhost:3306/db0?useUnicode=true&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=UTC

sharding.jdbc.datasource.db0.username=root

sharding.jdbc.datasource.db0.password=Aa123456

# 第二個數(shù)據(jù)庫

sharding.jdbc.datasource.db1.type=com.zaxxer.hikari.HikariDataSource

sharding.jdbc.datasource.db1.driver-class-name=com.mysql.cj.jdbc.Driver

sharding.jdbc.datasource.db1.jdbc-url=jdbc:mysql://localhost:3306/db1?useUnicode=true&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=UTC

sharding.jdbc.datasource.db1.username=root

sharding.jdbc.datasource.db1.password=Aa123456

# 第三個數(shù)據(jù)庫

sharding.jdbc.datasource.db2.type=com.zaxxer.hikari.HikariDataSource

sharding.jdbc.datasource.db2.driver-class-name=com.mysql.cj.jdbc.Driver

sharding.jdbc.datasource.db2.jdbc-url=jdbc:mysql://localhost:3306/db2?useUnicode=true&useJDBCCompliantTimezoneShift=true&useLegacyDatetimeCode=false&serverTimezone=UTC

sharding.jdbc.datasource.db2.username=root

sharding.jdbc.datasource.db2.password=Aa123456

# 水平拆分的數(shù)據(jù)庫(表) 配置分庫 + 分表策略 行表達(dá)式分片策略

# 分庫策略

sharding.jdbc.config.sharding.default-database-strategy.inline.sharding-column=id

sharding.jdbc.config.sharding.default-database-strategy.inline.algorithm-expression=db$->{id % 3}

# 分表策略 其中book為邏輯表 分表主要取決于id行

sharding.jdbc.config.sharding.tables.book.actual-data-nodes=db$->{0..2}.book_$->{0..2}

sharding.jdbc.config.sharding.tables.book.table-strategy.inline.sharding-column=count

# 分片算法表達(dá)式

sharding.jdbc.config.sharding.tables.book.table-strategy.inline.algorithm-expression=book_$->{count % 3}

# 主鍵 UUID 18位數(shù) 如果是分布式還要進(jìn)行一個設(shè)置 防止主鍵重復(fù)

#sharding.jdbc.config.sharding.tables.user.key-generator-column-name=id

# 打印執(zhí)行的數(shù)據(jù)庫以及語句

sharding.jdbc.config.props..sql.show=true

spring.main.allow-bean-definition-overriding=true

#讀寫分離

sharding.jdbc.datasource.dsmaster =

接口測試使用postman

示例:GET請求------>http://localhost:8080/book POST請求:------->http://localhost:8080/book?id=1&name=java編程思想&count=8

demo的github地址:https://github.com/Macky-He/spring-boot--shardingsphere-examples?

三智玻、總結(jié)

分庫分表實(shí)現(xiàn)按照官方文檔做一個demo是第一步,如需深入還需要研究源碼芙代,研究架構(gòu)吊奢,研究思想;此文僅作為入門demo搭建指南纹烹,如需深入理解页滚,還請移步至官方文檔,

官方文檔:https://shardingsphere.apache.org/document/current/cn/manual/sharding-jdbc/usage/sharding/

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末铺呵,一起剝皮案震驚了整個濱河市裹驰,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌片挂,老刑警劉巖幻林,帶你破解...
    沈念sama閱讀 211,042評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異音念,居然都是意外死亡沪饺,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,996評論 2 384
  • 文/潘曉璐 我一進(jìn)店門闷愤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來整葡,“玉大人,你說我怎么就攤上這事讥脐≡饩樱” “怎么了?”我有些...
    開封第一講書人閱讀 156,674評論 0 345
  • 文/不壞的土叔 我叫張陵旬渠,是天一觀的道長魏滚。 經(jīng)常有香客問我,道長坟漱,這世上最難降的妖魔是什么鼠次? 我笑而不...
    開封第一講書人閱讀 56,340評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上腥寇,老公的妹妹穿的比我還像新娘成翩。我一直安慰自己,他們只是感情好赦役,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,404評論 5 384
  • 文/花漫 我一把揭開白布麻敌。 她就那樣靜靜地躺著,像睡著了一般掂摔。 火紅的嫁衣襯著肌膚如雪术羔。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,749評論 1 289
  • 那天乙漓,我揣著相機(jī)與錄音级历,去河邊找鬼。 笑死叭披,一個胖子當(dāng)著我的面吹牛寥殖,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播涩蜘,決...
    沈念sama閱讀 38,902評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼嚼贡,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了同诫?” 一聲冷哼從身側(cè)響起粤策,我...
    開封第一講書人閱讀 37,662評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎误窖,沒想到半個月后掐场,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,110評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡贩猎,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,451評論 2 325
  • 正文 我和宋清朗相戀三年熊户,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片吭服。...
    茶點(diǎn)故事閱讀 38,577評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡嚷堡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出艇棕,到底是詐尸還是另有隱情蝌戒,我是刑警寧澤,帶...
    沈念sama閱讀 34,258評論 4 328
  • 正文 年R本政府宣布沼琉,位于F島的核電站北苟,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏打瘪。R本人自食惡果不足惜友鼻,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,848評論 3 312
  • 文/蒙蒙 一傻昙、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧彩扔,春花似錦妆档、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,726評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至敦捧,卻和暖如春须板,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背兢卵。 一陣腳步聲響...
    開封第一講書人閱讀 31,952評論 1 264
  • 我被黑心中介騙來泰國打工习瑰, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人济蝉。 一個月前我還...
    沈念sama閱讀 46,271評論 2 360
  • 正文 我出身青樓杰刽,卻偏偏與公主長得像菠发,于是被迫代替她去往敵國和親王滤。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,452評論 2 348