中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Mybatis深度整合Mysql的Json字段

發布時間:2020-06-23 16:59:03 來源:網絡 閱讀:10530 作者:wb59f817a362984 欄目:軟件技術

概述
以前當業務數據結構變化時,往往需要采用的方案是:
?修改表結構增加字段
?遇到數據結構有list結構時,新建1對多的關聯子表
?用字典表表示字段的增加
以上方案對代碼侵入性很強,同時與舊業務數據結構不兼容。導致代碼從實體類
、Dao、Service、Controller層都要修改。

隨著NOSQL數據庫的廣泛應用,可擴展的存儲方式在關系型數據庫中也有了很好的支持,最新的MySQL5.7中就新增加了一個數據類型JSON,使用mysql的json類型字段做擴展字段,可以以json串形式動態的存儲任意結構的數據,包括list結構的數據也不必再創建子表。代碼的實體類和Dao層不必修改,其他層代碼修改量也能夠減少。
Mybatis深度整合Mysql的Json字段

Mysql常見json字段操作
Mysql5.7開始支持json字段
創建帶有json字段的表micro_test,其中extcol為json類型字段

CREATE TABLE `micro_test` (
  `id` int(11) NOT NULL AUTO_INCREMENT,
  `meta_name` varchar(100) DEFAULT NULL COMMENT '元數據名稱',
  `create_time` datetime DEFAULT NULL COMMENT '創建時間',
  `update_time` datetime DEFAULT NULL COMMENT '更新時間',
  `extcol` json DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB CHARSET=utf8;

插入json字段
可按照json字符串插入json字段

Insert into micro_test (extcol,meta_name,create_time) 
values('{"name":"tomcat","age":15}',’123’,now());

查詢json字段
可以根據path查詢json字段中全部或部分數據
Select meta_name,extcol->>'$.name' as name,extcol->>'$.age' as age from micro_test;

修改json字段
可以根據path局部更新json字段中數據
Update micro_test set extcol=json_set(extcol,'$.name','jeffrey') where meta_name='123'

Mysql5.7.22版本以后支持JSON_MERGE_PATCH
可以省略path參數,全面更新json字段中數據

Update micro_test set extcol=json_set(extcol,'{“name”:”n1”,”age”:30}') where meta_name='123'

Mybatis使用Json字段
按照mybatis常規方式把json函數寫入到xml文件中的sql中,即可支持json字段增刪改查。但查詢出的json字段為字符串類型,需要手工轉成bean,插入時需手工把bean轉成json字符串,這樣做不利于面向對象編程。
Mybatis深度整合Mysql的Json字段

Mybatis深度整合Json字段
實現bean與json串在mybatis內部轉換,這樣做的優點是dao層代碼和sql不變,service層可以增刪改查不同的動態Entity對象。更符合面向對象編程習慣提高開發效率。
Mybatis深度整合Mysql的Json字段

Extcol開源項目實現Mybatis與mysql的json字段深度整合
項目地址為:
https://github.com/jeffreyning/extcol.git

pom引用extcol的jar

<dependency>
    <groupId>com.github.jeffreyning</groupId>
    <artifactId>extcol</artifactId>
    <version>0.0.1-RELEASE</version>
</dependency>

Extcol包中TypeHandler子類TagToJsonTypeHandler 實現mybatis在數據庫操作過程中的參數輸入和結果轉換的攔截。攔截父類為ExtBeanWrapper的對象。
使TagToJsonTypeHandler生效需要配置

mybatis-plus:
  typeHandlersPackage: com.nh.micro.ext.th

Extcol包中ExtBeanWrapper類,作為json對象轉換的目標對象,內有map成員變量保存實際數據,getobj和setobj方法是使用fastjson做對象與map的轉換。

使用Extcol的Demo
引入和配置好extcol后,在demo業務系統工程中編寫對應micro_test表的實體類TestDto,其中json字段的成員變量類型是ExtBeanWrapper。

public class TestDto  {
    private Integer id;
    private String metaKey;
    private String metaName;
    private String metaType;
    private Date createTime;

    private ExtBeanWrapper extcol;

    public ExtBeanWrapper getExtcol() {
        return extcol;
    }
    public void setExtcol(ExtBeanWrapper extcol) {
    }

擴展字段業務bean
例如擴展bean為ExtEntity有兩個在數據庫中json字段動態存儲的字段t1和t2

public class ExtEntity {
    private String t1;
    private String t2;
    public String getT1() {
        return t1;
    }
    public void setT1(String t1) {
        this.t1 = t1;
    }
    public String getT2() {
        return t2;
    }
    public void setT2(String t2) {
        this.t2 = t2;
    }
}

在以TestDto為更新和插入時的參數操作時,mybatis將負責將bean轉為json串

<update id="insertInfo4JsonXml" parameterType="com.test.nm.order.dto.TestDto">
        insert into micro_test(meta_name,extcol,create_time) values (#{metaName},#{extcol},now()) 
</update>

當執行查詢語句時,返回的結果映射到ExtBeanWrapper 類型的字段時,mybatis將負責將json串轉為ExtBeanWrapper ,且這個ExtBeanWrapper 可以按照不同的業務bean自適應轉化。

  <resultMap id="TestDto" type="com.test.nm.order.dto.TestDto" >
    <id column="id" property="id" jdbcType="INTEGER" />
    <result column="meta_name" property="metaName" jdbcType="VARCHAR" />
    <result column="create_time" property="createTime" jdbcType="TIMESTAMP" />
    <result column="extcol" property="extcol" jdbcType="VARCHAR" />
  </resultMap>

    <select id="getInfo4JsonXml" resultMap="TestDto" parameterType="java.lang.String">
        SELECT * from micro_test where meta_name=#{name}
    </select>

取查詢結果中JSON字段中存儲的業務類ExtEntity 的代碼

    public List<TestDto> testQuery4JsonXml(String name){
        List<TestDto> retList=testDao.getInfo4JsonXml(name);
        if(retList!=null){
            for(TestDto testDto:retList){
                ExtBeanWrapper extBeanWrapper=testDto.getExtcol();
                ExtEntity extEntity=(ExtEntity) extBeanWrapper.getObj(ExtEntity.class);
                System.out.println(extEntity.getT1());
            }
        }
        return retList;
    }

Mybatisplus的bean更新時使用JSON_MERGE_PATCH
修改mybatisplus的AutoSqlInjector

    private String getPlaceTag(String row){
        int start=row.indexOf("#{");
        int end=row.indexOf("}")+1;
        String temp=row.substring(start,end);
        System.out.println(temp);
        return temp;
    }
    private String getColTag(String row){
        int end=row.indexOf("=#{");
        int start=0;
        if(row.contains("<if")){
            start=row.indexOf(">")+1;
        }
        String temp=row.substring(start,end);
        System.out.println(temp);
        return temp;
    }
    private String createNewPlace(String colTag,String placeTag){
        String temp="json_merge_patch("+colTag+","+placeTag+")";
        return temp;
    }
    protected void injectUpdateByIdSql(boolean selective, Class<?> mapperClass, Class<?> modelClass, TableInfo table) {
        SqlMethod sqlMethod = selective ? SqlMethod.UPDATE_BY_ID : SqlMethod.UPDATE_ALL_COLUMN_BY_ID;
        String temp=sqlSet(selective, table, "et.");
        String osql=temp;
        if(selective){
            String[] tempArray=temp.split("\n\t");
            StringBuilder sb=new StringBuilder("");
            for(String row:tempArray){
                if(row.contains("typeHandler")){
                    System.out.println(getPlaceTag(row));
                    String placeTag=getPlaceTag(row);
                    System.out.println(getColTag(row));
                    String colTag=getColTag(row);
                    String nPlaceTag=createNewPlace(colTag, placeTag);
                    System.out.println(nPlaceTag);
                    row=row.replace(placeTag, nPlaceTag);
                    sb.append(row).append("\n\t");
                }else{
                    sb.append(row).append("\n\t");
                }
            }
            osql=sb.toString();
        }
        String sql = String.format(sqlMethod.getSql(), table.getTableName(), osql, table.getKeyColumn(),
                "et." + table.getKeyProperty(),
                "<if test=\"et instanceof java.util.Map\">"
                        + "<if test=\"et.MP_OPTLOCK_VERSION_ORIGINAL!=null\">"
                        + "and ${et.MP_OPTLOCK_VERSION_COLUMN}=#{et.MP_OPTLOCK_VERSION_ORIGINAL}"
                        + "</if>"
                        + "</if>"
        );
        System.out.println(sql);

        SqlSource sqlSource = languageDriver.createSqlSource(configuration, sql, modelClass);
        this.addUpdateMappedStatement(mapperClass, modelClass, sqlMethod.getMethod(), sqlSource);
    }
向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

淅川县| 通化市| 延寿县| 新绛县| 杨浦区| 鹤岗市| 绿春县| 古蔺县| 淮南市| 新宾| 岑巩县| 株洲县| 克东县| 贵州省| 夏津县| 麻栗坡县| 台北县| 泰州市| 清远市| 炉霍县| 潞城市| 金寨县| 冕宁县| 女性| 承德县| 烟台市| 安乡县| 顺昌县| 儋州市| 辉县市| 丰镇市| 阿荣旗| 岳普湖县| 临漳县| 保定市| 河间市| 简阳市| 鹿邑县| 扬中市| 布拖县| 介休市|