欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

利用Mybatis向PostgreSQL中插入并查詢JSON字段

 更新時間:2022年07月22日 14:59:59   作者:T.Y.Bao  
這篇文章主要介紹了利用Mybatis向PostgreSQL中插入并查詢JSON字段,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下

前言:

這里我使用的是TimescaleDB,加了一個時間戳字段,不過沒差。關(guān)于PostgreSQL中Json數(shù)據(jù)類型的操作,可以參考官網(wǎng)。

應(yīng)用場景介紹

將TCP發(fā)過來的數(shù)據(jù)包(通過消息隊列發(fā)過來)解析出數(shù)據(jù)(一個數(shù)據(jù)包含有多幀,一幀中含有多條信息),并和本地規(guī)則表的格式對應(yīng)起來。以JsonLineMsg實體類代表對應(yīng)的一幀數(shù)據(jù):

package tsdb.entity;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import lombok.ToString;
import java.sql.Timestamp;
@Data
@AllArgsConstructor
@NoArgsConstructor
@ToString
public class JsonLineMsg {
    private Timestamp timeStamp; // 時間戳

    private String keyAndRuleData; // key value,key為根據(jù)規(guī)則表生成的唯一標(biāo)識,value為TCP解析出的對應(yīng)的數(shù)據(jù)。這個字段對應(yīng)數(shù)據(jù)庫中的Json類型字段,String類型進(jìn)入數(shù)據(jù)庫還需轉(zhuǎn)換為Json格式。
}

對應(yīng)psql的表結(jié)構(gòu)為:

上面JsonLineMsg實體類的一個對象就代表的一幀中的所有數(shù)據(jù)項many(key:value)keyAndRuleData字段用來存儲所有數(shù)據(jù)項,在psql中對應(yīng)一個類型為json(或jsonb)的字段。

數(shù)據(jù)insert

為了查詢JSON中的字段,在insert的過程中有些注意事項,如果插入時JSON格式不正確,查詢JSON字段是總返回null。

記錄一下:為了降低數(shù)據(jù)庫打開關(guān)閉的耗時,每積累20幀持久化一次。

note:

  • foreach批量插入 、 mybatis ExecutorType.BATCH模式插入 、 for循環(huán)insert
  • 其實實際意義上來說,包括在程序里面for循環(huán)還是在sql里面for循環(huán)都不算是批量操作。只有將ExecutorType設(shè)置為BATCH模式才是真正意義上的批量操作。 并且事實證明在sql循環(huán)時設(shè)置batch與否其實執(zhí)行時間差別不是很大,幾乎可以忽略不計。所以其實如果不是特別要求性能??梢灾苯釉趕ql中使用for循環(huán)即可 。謹(jǐn)慎使用batch,如果需要使用batch,請在需要的函數(shù)上面設(shè)置batch,不要全局使用。因為batch也是有副作用的。 比如在Insert操作時,在事務(wù)沒有提交之前,是沒有辦法獲取到自增的id,;此外,對于update、delete無法返回更新、插入條數(shù)。這在某型情形下是不符合業(yè)務(wù)要求的。上面的是搬運的,不過后來有看了看,還是應(yīng)該用BATCH的Executor來批量導(dǎo)入,實際項目中foreach不可控,指不定啥時候就報錯了,文章最后記錄了ExecutorType為BATCH寫法的關(guān)鍵部分) foreach的xml拼接sql是最不推薦的方式,使用時有大段的xml和sql語句要寫,很容易出錯,工作效率很低。更關(guān)鍵點是,雖然效率尚可,但是真正需要效率的時候你掛了,要你何用? 批處理執(zhí)行是有大數(shù)據(jù)量插入時推薦的做法,使用起來也比較方便。
  • 關(guān)于批處理的方式的具體說明,可以參考推文強烈推薦MyBatis 三種批量插入方式的比較或者去StackOverFlow查一下,講解的比較全面,總之,還是用ExecutorType為BATCH寫法比較靠譜。

一幀中包含多條信息,一條信息對應(yīng)一個key:value,所以每次從規(guī)則表生成的key和TCP解析出的value都要加到一個代表一幀所有數(shù)據(jù)的JSON串中。

要注意的代碼如下:

                 // 存儲一幀的所有key:value
                StringBuilder json = new StringBuilder();
                json.append("{");
                // frmLen 幀中信息個數(shù)
                for (int j = 0; j < frmLen; j++) {
                    StatRule stat = frm.getStat(j);
                    assert stat != null;
                    // 一條stat的key和value
                    int key = stat.getKey();
                    long value = System.nanoTime();
//                    String value = ParseStat.Parse(datas, stat);
                    json.append("\"");  // key左右必須加引號,key必為String類型
                    json.append(key);
                    json.append("\"");
                    json.append(":");
//                    json.append("\"");
                    json.append(value); // value左右不是必須加引號,若是String則加
//                    json.append("\"");
                    if ((j != statLen - 1)) {
                        json.append(",");
                    }
                }
                json.append("}");
                JsonLineMsg jsonLineMsg = new JsonLineMsg(new Timestamp(System.currentTimeMillis()), json.toString());

要注意的就是這個keyvalue加入數(shù)據(jù)庫的類型如果為text(即java字符串)就要加引號,所以key兩頭必須加,value看情況。

對應(yīng)的XML中的語句:

    <insert id="batchInsertJsonLineMsg"
            useGeneratedKeys="true" >
        insert into jsonlinemsg (timestamp ,keyandruledata ) values
        <foreach item="item" collection="list" separator="," close=";">
            (#{item.timeStamp},(#{item.keyAndRuleData})::json)
        </foreach>
    </insert>

這個::json就是將非json類型轉(zhuǎn)為json類型,否則JAVA中String類型會對應(yīng)其他的數(shù)據(jù)庫字段類型,插入會報錯。

note: psql 4種類型轉(zhuǎn)換 https://www.postgresql.org/docs/14/sql-syntax-lexical.html

  • type 'string' 只能用于字面常量轉(zhuǎn)換、且不能用于數(shù)組中
  • typename ( 'string' ) 可用于運行時類型轉(zhuǎn)換
  • 'string'::type 可用于數(shù)組,可用于運行時類型轉(zhuǎn)換
  • CAST ( 'string' AS type ) 可用于數(shù)組,可用于運行時類型轉(zhuǎn)換

插入后用Navicat查看:

如果查看到類似于 "{"1":"1_234"}"{\"1\":\"1_123\"}這樣,格式就是不正確的,查詢JSON中字段會返回null。

數(shù)據(jù)select

  <select id="selectValueData" resultType="String">
        select keyandruledata::json ->>#{key}  from jsonlinemsg where timestamp = (#{time}::timestamp)
    </select>

要注意的就是這個::json,至于 -> 還是 ->>可以參考開頭的官網(wǎng)鏈接。

ps: timescaledb官網(wǎng)推薦用jsonb,但是我測試發(fā)現(xiàn)jsonb查詢插入都比不上json,不知道為啥
ps: 發(fā)現(xiàn)了,原來是轉(zhuǎn)換為tsdb時,索引沒建立起來,重新建表又測試了一遍,確實jsonb讀取快。

BATCH 批量插入

// 獲取連接的方法,設(shè)置ExecutorType.BATCH以及關(guān)閉自動提交
    public static SqlSession getSessionForBatch(String xmlPath, Properties properties) throws IOException {
        return MybatisUtil.getSqlSessionFactory(xmlPath, properties).openSession(ExecutorType.BATCH,false);
    }
    public void update(List<PropUrl> propUrlLst) throws IOException {
        // ExecutorType.BATCH
        try (SqlSession session = MybatisUtil.getSessionForBatch(myBatisConfigXmlPath)) {
            InitTsdbUrlTableMapper mapper = (InitTsdbUrlTableMapper) session.getMapper(mapperClazz);

            for (int i = 0; i < propUrlLst.size(); i++) {
                mapper.updatePropMatchRule(propUrlLst.get(i));
                // 每50次提交一次防止內(nèi)存溢出
                if ((i+1) % 50 == 0) {
                    session.commit();
                    session.clearCache();
                }
            }
            session.commit();
            session.clearCache();

            log.info("update successfully ->{}", propUrlLst);
        }
    }

到此這篇關(guān)于利用Mybatis向PostgreSQL中插入并查詢JSON字段的文章就介紹到這了,更多相關(guān)Mybatis查詢JSON字段內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Java中Lambda表達(dá)式基礎(chǔ)及使用

    Java中Lambda表達(dá)式基礎(chǔ)及使用

    這篇文章主要介紹了Lambda 是JDK 8 的重要新特性。它允許把函數(shù)作為一個方法的參數(shù)(函數(shù)作為參數(shù)傳遞進(jìn)方法中),使用 Lambda 表達(dá)式可以使代碼變的更加簡潔緊湊,使Java代碼更加優(yōu)雅,感興趣的小伙伴一起來學(xué)習(xí)吧
    2021-08-08
  • SpringBoot Admin使用及心跳檢測原理分析

    SpringBoot Admin使用及心跳檢測原理分析

    這篇文章主要介紹了SpringBoot Admin使用及心跳檢測原理分析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • Java Web 簡單的分頁顯示實例代碼

    Java Web 簡單的分頁顯示實例代碼

    這篇文章主要介紹了Java Web 簡單的分頁顯示實例代碼的相關(guān)資料,本文通過,計算總的頁數(shù)和查詢指定頁數(shù)據(jù)兩個方法實現(xiàn)分頁效果,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-06-06
  • Java為何需要平衡方法調(diào)用與內(nèi)聯(lián)

    Java為何需要平衡方法調(diào)用與內(nèi)聯(lián)

    這篇文章主要介紹了Java為何需要平衡方法調(diào)用與內(nèi)聯(lián),幫助大家更好的理解和使用Java,感興趣的朋友可以了解下
    2021-01-01
  • SpringBoot集合Mybatis過程解析

    SpringBoot集合Mybatis過程解析

    這篇文章主要介紹了SpringBoot集合Mybatis過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-10-10
  • SpringBoot攔截器讀取流后不能再讀取的問題

    SpringBoot攔截器讀取流后不能再讀取的問題

    這篇文章主要介紹了SpringBoot攔截器讀取流后不能再讀取的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • Java實現(xiàn)的AES256加密解密功能示例

    Java實現(xiàn)的AES256加密解密功能示例

    這篇文章主要介紹了Java實現(xiàn)的AES256加密解密功能,結(jié)合完整實例形式分析了Java實現(xiàn)AES256加密解密功能的步驟與相關(guān)操作技巧,需要的朋友可以參考下
    2017-02-02
  • Springboot實現(xiàn)多服務(wù)器session共享

    Springboot實現(xiàn)多服務(wù)器session共享

    這篇文章主要為大家詳細(xì)介紹了Springboot實現(xiàn)多服務(wù)器session共享,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-05-05
  • 詳解Java利用同步塊synchronized()保證并發(fā)安全

    詳解Java利用同步塊synchronized()保證并發(fā)安全

    這篇文章主要介紹了Java利用同步塊synchronized()保證并發(fā)安全,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03
  • java實現(xiàn)簡單聊天室單人版

    java實現(xiàn)簡單聊天室單人版

    這篇文章主要為大家詳細(xì)介紹了java實現(xiàn)簡單聊天室的單人版,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-07-07

最新評論