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

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

Fluent MyBatis實現動態SQL

瀏覽:27日期:2023-10-18 11:35:34
目錄數據準備代碼生成在 WHERE 條件中使用動態條件在 UPDATE 使用動態更新choose 標簽參考

MyBatis 令人喜歡的一大特性就是動態 SQL。在使用 JDBC 的過程中, 根據條件進行 SQL 的拼接是很麻煩且很容易出錯的,MyBatis雖然提供了動態拼裝的能力,但這些寫xml文件,也確實折磨開發。Fluent MyBatis提供了更貼合Java語言特質的,對程序員友好的Fluent拼裝能力。

Fluent MyBatis動態SQL,寫SQL更爽

數據準備

為了后面的演示, 創建了一個 Maven 項目 fluent-mybatis-dynamic, 創建了對應的數據庫和表

DROP TABLE IF EXISTS `student`;CREATE TABLE `student`( `id` bigint(21) unsigned NOT NULL AUTO_INCREMENT COMMENT ’編號’, `name` varchar(20) DEFAULT NULL COMMENT ’姓名’, `phone`varchar(20) DEFAULT NULL COMMENT ’電話’, `email`varchar(50) DEFAULT NULL COMMENT ’郵箱’, `gender` tinyint(2) DEFAULT NULL COMMENT ’性別’, `locked` tinyint(2) DEFAULT NULL COMMENT ’狀態(0:正常,1:鎖定)’, `gmt_created` datetime DEFAULT CURRENT_TIMESTAMP COMMENT ’存入數據庫的時間’, `gmt_modified` datetime DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT ’修改的時間’, `is_deleted` tinyint(2) DEFAULT 0, PRIMARY KEY (`id`)) ENGINE = InnoDB DEFAULT CHARSET = utf8mb4 COLLATE = utf8mb4_0900_ai_ci COMMENT =’學生表’;代碼生成

使用Fluent Mybatis代碼生成器,生成對應的Entity文件

public class Generator { static final String url = 'jdbc:mysql://localhost:3306/fluent_mybatis?useSSL=false&useUnicode=true&characterEncoding=utf-8'; /** * 生成代碼的package路徑 */ static final String basePackage = 'cn.org.fluent.mybatis.dynamic'; /** * 使用 test/resource/init.sql文件自動初始化測試數據庫 */ @BeforeAll static void runDbScript() {DataSourceCreatorFactory.create('dataSource'); } @Test void test() {FileGenerator.build(Nothing.class); } @Tables(/** 數據庫連接信息 **/url = url, username = 'root', password = 'password',/** Entity類parent package路徑 **/basePack = basePackage,/** Entity代碼源目錄 **/srcDir = 'src/main/java',/** 如果表定義記錄創建,記錄修改,邏輯刪除字段 **/gmtCreated = 'gmt_created', gmtModified = 'gmt_modified', logicDeleted = 'is_deleted',/** 需要生成文件的表 ( 表名稱:對應的Entity名稱 ) **/tables = @Table(value = {'student'}) ) public static class Nothing { }}

編譯項目,ok,下面我們開始動態SQL構造旅程

在 WHERE 條件中使用動態條件

在mybatis中,if 標簽是大家最常使用的。在查詢、刪除、更新的時候結合 test 屬性聯合使用。

示例:根據輸入的學生信息進行條件檢索

當只輸入用戶名時, 使用用戶名進行模糊檢索; 當只輸入性別時, 使用性別進行完全匹配 當用戶名和性別都存在時, 用這兩個條件進行查詢匹配查詢

mybatis動態 SQL寫法

<select resultMap='BaseResultMap' parameterType='com.homejim.mybatis.entity.Student'> select <include refid='Base_Column_List' /> from student <where><if test='name != null and name !=’’'> and name like concat(’%’, #{name}, ’%’)</if><if test='sex != null'> and sex=#{sex}</if> </where></select>

fluent mybatis動態寫法

@Repositorypublic class StudentDaoImpl extends StudentBaseDao implements StudentDao { /** * 根據輸入的學生信息進行條件檢索 * 1. 當只輸入用戶名時, 使用用戶名進行模糊檢索; * 2. 當只輸入性別時, 使用性別進行完全匹配 * 3. 當用戶名和性別都存在時, 用這兩個條件進行查詢匹配的用 * * @param name 姓名,模糊匹配 * @param isMale 性別 * @return */ @Override public List<StudentEntity> selectByNameOrEmail(String name, Boolean isMale) {return super.defaultQuery() .where.name().like(name, If::notBlank) .and.gender().eq(isMale, If::notNull).end() .execute(super::listEntity); }}

FluentMyBatis的實現方式至少有下面的好處

邏輯就在方法實現上,不需要額外維護xml,割裂開來 所有的編碼通過IDE智能提示,沒有字符串魔法值編碼 編譯檢查,拼寫錯誤能立即發現

測試

@SpringBootTest(classes = AppMain.class)public class StudentDaoImplTest extends Test4J { @Autowired StudentDao studentDao; @DisplayName('只有名字時的查詢') @Test void selectByNameOrEmail_onlyName() {studentDao.selectByNameOrEmail('明', null);// 驗證執行的sql語句db.sqlList().wantFirstSql().eq('' +'SELECT id, gmt_created, gmt_modified, is_deleted, email, gender, locked, name, phone ' +'FROM student ' +'WHERE name LIKE ?', StringMode.SameAsSpace);// 驗證sql參數db.sqlList().wantFirstPara().eqReflect(new Object[]{'%明%'}); } @DisplayName('只有性別時的查詢') @Test void selectByNameOrEmail_onlyGender() {studentDao.selectByNameOrEmail(null, false);// 驗證執行的sql語句db.sqlList().wantFirstSql().eq('' +'SELECT id, gmt_created, gmt_modified, is_deleted, email, gender, locked, name, phone ' +'FROM student ' +'WHERE gender = ?', StringMode.SameAsSpace);// 驗證sql參數db.sqlList().wantFirstPara().eqReflect(new Object[]{false}); } @DisplayName('姓名和性別同時存在的查詢') @Test void selectByNameOrEmail_both() {studentDao.selectByNameOrEmail('明', false);// 驗證執行的sql語句db.sqlList().wantFirstSql().eq('' +'SELECT id, gmt_created, gmt_modified, is_deleted, email, gender, locked, name, phone ' +'FROM student ' +'WHERE name LIKE ? ' +'AND gender = ?', StringMode.SameAsSpace);// 驗證sql參數db.sqlList().wantFirstPara().eqReflect(new Object[]{'%明%', false}); }}在 UPDATE 使用動態更新

只更新有變化的字段, 空值不更新

mybatis xml寫法

<update parameterType='...'> update student <set> <if test='name != null'>`name` = #{name,jdbcType=VARCHAR}, </if> <if test='phone != null'>phone = #{phone,jdbcType=VARCHAR}, </if> <if test='email != null'>email = #{email,jdbcType=VARCHAR}, </if> <if test='gender != null'>gender = #{gender,jdbcType=TINYINT}, </if> <if test='gmtModified != null'>gmt_modified = #{gmtModified,jdbcType=TIMESTAMP}, </if> </set> where id = #{id,jdbcType=INTEGER}</update>

fluent mybatis實現

@Repositorypublic class StudentDaoImpl extends StudentBaseDao implements StudentDao { /** * 根據主鍵更新非空屬性 * * @param student * @return */ @Override public int updateByPrimaryKeySelective(StudentEntity student) {return super.defaultUpdater() .update.name().is(student.getName(), If::notBlank) .set.phone().is(student.getPhone(), If::notBlank) .set.email().is(student.getEmail(), If::notBlank) .set.gender().is(student.getGender(), If::notNull) .end() .where.id().eq(student.getId()).end() .execute(super::updateBy); } }

測試

@SpringBootTest(classes = AppMain.class)public class StudentDaoImplTest extends Test4J { @Autowired StudentDao studentDao; @Test void updateByPrimaryKeySelective() {StudentEntity student = new StudentEntity() .setId(1L) .setName('test') .setPhone('13866668888');studentDao.updateByPrimaryKeySelective(student);// 驗證執行的sql語句db.sqlList().wantFirstSql().eq('' +'UPDATE student ' +'SET gmt_modified = now(), ' +'name = ?, ' +'phone = ? ' +'WHERE id = ?', StringMode.SameAsSpace);// 驗證sql參數db.sqlList().wantFirstPara().eqReflect(new Object[]{'test', '13866668888', 1L}); }}choose 標簽

在mybatis中choose when otherwise 標簽可以幫我們實現 if else 的邏輯。

查詢條件,假設 name 具有唯一性, 查詢一個學生

當 id 有值時, 使用 id 進行查詢; 當 id 沒有值時, 使用 name 進行查詢; 否則返回空

mybatis xml實現

<select resultMap='BaseResultMap' parameterType='...'> select <include refid='Base_Column_List' /> from student <where><choose> <when test='id != null'> and id=#{id} </when> <when test='name != null and name != ’’'> and name=#{name} </when> <otherwise> and 1=2 </otherwise></choose> </where></select>

fluent mybatis實現方式

@Repositorypublic class StudentDaoImpl extends StudentBaseDao implements StudentDao { /** * 1. 當 id 有值時, 使用 id 進行查詢; * 2. 當 id 沒有值時, 使用 name 進行查詢; * 3. 否則返回空 */ @Override public StudentEntity selectByIdOrName(StudentEntity student) { return super.defaultQuery() .where.id().eq(student.getId(), If::notNull) .and.name().eq(student.getName(), name -> isNull(student.getId()) && notBlank(name)) .and.apply('1=2', () -> isNull(student.getId()) && isBlank(student.getName())) .end() .execute(super::findOne).orElse(null); }}

測試

@SpringBootTest(classes = AppMain.class)public class StudentDaoImplTest extends Test4J { @Autowired StudentDao studentDao; @DisplayName('有 ID 則根據 ID 獲取') @Test void selectByIdOrName_byId() {StudentEntity student = new StudentEntity();student.setName('小飛機');student.setId(1L);StudentEntity result = studentDao.selectByIdOrName(student);// 驗證執行的sql語句db.sqlList().wantFirstSql().eq('' +'SELECT id, gmt_created, gmt_modified, is_deleted, email, gender, locked, name, phone ' +'FROM student ' +'WHERE id = ?', StringMode.SameAsSpace);// 驗證sql參數db.sqlList().wantFirstPara().eqReflect(new Object[]{1L}); } @DisplayName('沒有 ID 則根據 name 獲取') @Test void selectByIdOrName_byName() {StudentEntity student = new StudentEntity();student.setName('小飛機');student.setId(null);StudentEntity result = studentDao.selectByIdOrName(student);// 驗證執行的sql語句db.sqlList().wantFirstSql().eq('' +'SELECT id, gmt_created, gmt_modified, is_deleted, email, gender, locked, name, phone ' +'FROM student ' +'WHERE name = ?', StringMode.SameAsSpace);// 驗證sql參數db.sqlList().wantFirstPara().eqReflect(new Object[]{'小飛機'}); } @DisplayName('沒有 ID 和 name, 返回 null') @Test void selectByIdOrName_null() {StudentEntity student = new StudentEntity();StudentEntity result = studentDao.selectByIdOrName(student);// 驗證執行的sql語句db.sqlList().wantFirstSql().eq('' +'SELECT id, gmt_created, gmt_modified, is_deleted, email, gender, locked, name, phone ' +'FROM student ' +'WHERE 1=2', StringMode.SameAsSpace);// 驗證sql參數db.sqlList().wantFirstPara().eqReflect(new Object[]{}); }}參考

示例代碼地址Fluent MyBatis地址Fluent MyBatis文檔Test4J框架

到此這篇關于Fluent MyBatis實現動態SQL的文章就介紹到這了,更多相關Fluent MyBatis 動態SQL內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Mybatis 數據庫
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
欧美特黄一级大片| 欧美日韩一区二区国产| 亚洲精品在线a| 中文在线一区| 亚洲专区视频| 国产精品片aa在线观看| 日韩av免费大片| 桃色av一区二区| 另类中文字幕国产精品| 午夜在线一区二区| 青青草国产成人99久久| 国产成年精品| 免费黄色成人| 日韩激情精品| 精品国产第一福利网站| 亚洲深爱激情| 亚洲一区二区免费看| 免费成人在线影院| 久久久精品国产**网站| 国产精品原创| 欧洲av不卡| 欧美亚洲激情| 日韩亚洲国产欧美| 亚洲精品极品| 麻豆精品少妇| 99久久久久国产精品| 日本天堂一区| 国产精品丝袜在线播放| 国精品产品一区| 激情欧美日韩一区| 中文字幕免费精品| 麻豆国产精品视频| 亚洲精品一级二级| 免费看黄色91| 久久不卡国产精品一区二区| 久久不射中文字幕| 日韩在线成人| 国产精品国码视频| 裤袜国产欧美精品一区| 日韩一级精品| 国产精品白丝久久av网站| 五月激情久久| 天海翼亚洲一区二区三区| 国产精品jk白丝蜜臀av小说| av一区在线| 日本大胆欧美人术艺术动态| 国产精品麻豆成人av电影艾秋| 色在线视频观看| 蜜臀av性久久久久蜜臀aⅴ四虎| 国产精品中文| 午夜电影亚洲| 日韩欧美午夜| 日韩一级网站| 欧美一级一区| 日韩三区在线| 日本va欧美va精品发布| 高清av一区二区三区| 亚洲精品欧美| 国产美女高潮在线观看| 亚洲精品成人一区| 首页国产精品| 日韩中文字幕在线一区| 日韩免费福利视频| 欧美亚洲综合视频| 激情91久久| 精品亚洲a∨| 国产欧美大片| 1000部精品久久久久久久久| 日韩中出av| 一区二区小说| 久久一区视频| 久久亚洲风情| 粉嫩av一区二区三区四区五区 | 日韩深夜视频| 欧美在线91| 国产精品三上| 中文av在线全新| 国产欧美日韩精品高清二区综合区| 999在线观看精品免费不卡网站| 国产精品久久观看| 日韩国产在线一| 欧美日韩国产精品一区二区亚洲| 精品国产午夜| 日韩av一区二区在线影视| 欧美精品激情| 成人久久久久| 国产一区二区三区四区五区| 欧美日韩伊人| 中文字幕av一区二区三区四区| 性欧美xxxx免费岛国不卡电影| 九九99久久精品在免费线bt| 日韩精品高清不卡| 亚洲中午字幕| 欧美一级精品| 国产亚洲一区二区手机在线观看| 7777精品| 婷婷亚洲精品| 久久最新视频| 尤物在线精品| 亚洲香蕉网站| 99久久九九| 欧美日韩一二三四| 久久青青视频| 国产一区三区在线播放| 欧美极品一区二区三区| 国产欧美在线| 欧美影院视频| 国产欧美日韩一区二区三区在线| 亚洲一区二区免费在线观看| 在线综合欧美| 国户精品久久久久久久久久久不卡 | 涩涩涩久久久成人精品| 亚洲欧美日本视频在线观看| 在线日韩电影| 亚洲国产专区| 欧美丝袜一区| 蜜桃国内精品久久久久软件9| 91精品国产乱码久久久久久久 | 国产精品成人国产| 国产精品theporn| 麻豆国产精品| 国产一区二区三区91| 精品久久国产一区| 高清一区二区三区av| 毛片在线网站| 99国产精品免费视频观看| 久久久久中文| 激情综合亚洲| 尤物精品在线| 亚洲三级av| 亚洲不卡视频| 欧美午夜三级| 精品中文在线| 麻豆精品蜜桃| 91成人精品视频| 国产精品色网| 日韩综合小视频| 国产精品久久久久久模特| 国产极品久久久久久久久波多结野| 欧美精品99| 久久久精品国产**网站| 久久影院午夜精品| 国内精品福利| 美国三级日本三级久久99 | 91午夜精品| 久久亚洲道色| 亚洲成人不卡| 一区二区视频欧美| 亚洲+小说+欧美+激情+另类| 国产精品观看| 日韩欧美一区二区三区在线视频| 99久久精品费精品国产| 在线午夜精品| 日本强好片久久久久久aaa| 麻豆一区二区三区| 欧美羞羞视频| 午夜在线视频一区二区区别| 日韩av在线免费观看不卡| 精品香蕉视频| 一级欧洲+日本+国产| 久久国产精品99国产| 久久国内精品| 日本不卡的三区四区五区| 国产精品s色| 久久久夜精品| 综合一区二区三区| 日韩av黄色在线| 国产剧情在线观看一区| 欧美国产美女| 久久国产99| 国产精品jk白丝蜜臀av小说| 久久精品国语| 日本国产欧美| 久久久一本精品| 日韩亚洲精品在线观看| 精品三区视频| 欧美一级精品| 欧美日韩99| 欧美日韩一二三四| 婷婷亚洲成人| av高清一区| 日本综合精品一区| 精品一区二区三区中文字幕视频 | 日韩一区精品| 免播放器亚洲一区| 国产精品观看| 99精品电影| 日本在线一区二区三区| 九九九精品视频| 国产精品试看| 国产成人精品一区二区免费看京 | 精品久久美女| 在线国产日韩| 精品三级久久久| 手机精品视频在线观看| 精品日本视频| 午夜久久av| 欧美一区久久久| 日韩在线视频一区二区三区| 成人在线黄色|