mysql中json類型字段的基本用法實(shí)例
前言
mysql從5.7.8版本開始原生支持了JSON類型數(shù)據(jù),同時(shí)可以對(duì)JSON類型字段中的特定的值進(jìn)行查詢和更新等操作,通過(guò)增加JSON類型的屬性可以大大的提高我們?cè)趍ysql表中存儲(chǔ)的數(shù)據(jù)的拓展性,無(wú)需每次新增字段時(shí)都進(jìn)行表結(jié)構(gòu)的調(diào)整,下面我們不深入講解底層的實(shí)現(xiàn)原理,我們主要來(lái)梳理一下我們?cè)谌粘9ぷ髦惺褂脤?shí)踐
基本環(huán)境
mysql版本:8.0.28
springboot版本: 2.2.2
測(cè)試表結(jié)構(gòu):
CREATE TABLE `t_json` ( `id` int unsigned NOT NULL AUTO_INCREMENT, `name` varchar(100) DEFAULT NULL, `json_obj` json DEFAULT NULL, `create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP, `update_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP, PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci;
表結(jié)構(gòu)對(duì)應(yīng)的實(shí)體類:
@Data
public class JsonTest {
private Integer id;
private String name;
private JsonObj jsonObj;
}JsonObj類
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class JsonObj {
private String data;
private int age;
}自定義格式轉(zhuǎn)換類:轉(zhuǎn)換過(guò)程通過(guò)fastjson來(lái)進(jìn)行,需依賴fastjson對(duì)應(yīng)的pom文件
import com.alibaba.fastjson.JSON;
import lombok.extern.slf4j.Slf4j;
import org.apache.ibatis.type.BaseTypeHandler;
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.MappedTypes;
import java.sql.CallableStatement;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
@MappedTypes(String.class)
@Slf4j
public class JsonTypeHandler<T extends Object> extends BaseTypeHandler<T> {
private Class<T> clazz;
public JsonTypeHandler(Class<T> clazz) {
if (clazz == null) {
throw new IllegalArgumentException("Type argument cannot be null");
}
this.clazz = clazz;
}
@Override
public void setNonNullParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException {
ps.setString(i, this.toJson(parameter));
}
@Override
public T getNullableResult(ResultSet rs, String columnName) throws SQLException {
return this.toObject(rs.getString(columnName), clazz);
}
@Override
public T getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
return this.toObject(rs.getString(columnIndex), clazz);
}
@Override
public T getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
return this.toObject(cs.getString(columnIndex), clazz);
}
private String toJson(T object) {
return JSON.toJSONString(object);
}
private T toObject(String content, Class<?> clazz) {
if (content != null && !content.isEmpty()) {
return (T) JSON.parseObject(content, clazz);
} else {
return null;
}
}
}JSON類型字段常用操作
插入JSON類型數(shù)據(jù)
在項(xiàng)目開發(fā)中,表結(jié)構(gòu)中的json字段可以用JSONObject這樣的比較靈活的方式來(lái)傳遞,但是這樣的方式有一個(gè)比較大的問(wèn)題就是我們?cè)讷@得這個(gè)結(jié)構(gòu)后無(wú)法很直觀的確定json字段中存儲(chǔ)的數(shù)據(jù),比較好的一種方式是我們將表結(jié)構(gòu)中的json類型的字段以自定義的類來(lái)存儲(chǔ),這樣我們?cè)俅嫒∵@個(gè)類對(duì)應(yīng)的對(duì)象的時(shí)候,就可以明確的知道數(shù)據(jù)庫(kù)中對(duì)應(yīng)的json字段到底存儲(chǔ)的是一些什么樣的key,如果需要進(jìn)行調(diào)整的話,我們只需要在該類中新增新的字段即可,完全無(wú)需對(duì)數(shù)據(jù)庫(kù)進(jìn)行任何的調(diào)整。這樣的存儲(chǔ)方式我們?cè)俨迦牒筒樵冊(cè)撟侄蔚臅r(shí)候需要指定一個(gè)指定的數(shù)據(jù)類型轉(zhuǎn)換的類來(lái)對(duì)數(shù)據(jù)庫(kù)中的JSON格式數(shù)據(jù)和我們?cè)陧?xiàng)目中的自定義類進(jìn)行轉(zhuǎn)換的類,具體如下圖所示:
@Mapper
public interface JsonMapper {
@Insert({
"insert into t_json set name= #{name}, json_obj = #{jsonObj ,jdbcType=OTHER, typeHandler=cn.example.eureka.service.one.config.JsonTypeHandler}"
})
int insert(JsonTest json);
}查詢JSON類型數(shù)據(jù)
在進(jìn)行查詢時(shí),由于也需要進(jìn)行json格式數(shù)據(jù)和自定義類的轉(zhuǎn)換,所以我們需要指定對(duì)應(yīng)的json字段和轉(zhuǎn)換的工具類,通過(guò)@Result注解來(lái)進(jìn)行指定
@Mapper
public interface JsonMapper {
@Select({"<script>",
"select * from t_json where id = #{id}",
"</script>"})
@Results(
@Result(column = "json_obj", property = "jsonObj", typeHandler = JsonTypeHandler.class)
)
JsonTest getById(Integer id);
}更新JSON類型數(shù)據(jù)中的特定字段
進(jìn)行指定字段的更新的話,有兩種方式可以采用,一種方式是先將該json格式字段中的所有數(shù)據(jù)都取出,然后通過(guò)修改當(dāng)前對(duì)象的值,然后整個(gè)json格式字段set進(jìn)去;第二種方式直接通過(guò)json格式的特定SQL語(yǔ)法來(lái)進(jìn)行指定key的更新;下面的例子里面我們分別根據(jù)這兩種不同的模式進(jìn)行更新操作
//模式一:整體更新整個(gè)json字段
@Update({
"update t_json set json_obj = #{jsonObj ,jdbcType=OTHER, typeHandler=cn.example.eureka.service.one.config.JsonTypeHandler} where id = #{id}"
})
int update(JsonTest jsonTest);
//模式二:只更新json字段中的特定key
@Update({
"update t_json set json_obj = JSON_SET(json_obj, '$.data', #{data}) where id = #{id}"
})
int updateData(@Param("id") Integer id, @Param("data") String data);說(shuō)明
和上面的JSON_SET同樣可以用于修改的操作函數(shù)還有:JSON_INSERT、 JSON_REPLACE 、 JSON_REMOVE等,下面簡(jiǎn)單說(shuō)一下這幾個(gè)函數(shù)的不同
- JSON_INSERT : json字段中的key不存在才會(huì)修改
- JSON_REPLACE : json字段中的key存在才會(huì)修改
- JSON_SET : json字段中的可以不管是否存在都會(huì)修改
- JSON_REMOVE : 移除json字段中的指定key
如下所示JSON_REMOVE的用法:
@Update({
"update t_json set json_obj = JSON_REMOVE(json_obj, '$.age') where id = #{id}"
})
int removeAge(@Param("id") Integer id);匹配JSON類型數(shù)據(jù)中的特定字段
//模式一
@Select({
"select * from t_json where json_obj -> '$.age' = #{age}"
})
@Results(
@Result(column = "json_obj", property = "jsonObj", typeHandler = JsonTypeHandler.class)
)
List<JsonTest> getByAge(int age);
//模式二
@Select({
"select * from t_json where JSON_EXTRACT(json_obj , '$.data') = #{data}"
})
@Results(
@Result(column = "json_obj", property = "jsonObj", typeHandler = JsonTypeHandler.class)
)
List<JsonTest> getByData(String data);結(jié)語(yǔ)
到此這篇關(guān)于mysql中json類型字段的基本用法的文章就介紹到這了,更多相關(guān)mysql json類型字段用法內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MYSQL必知必會(huì)讀書筆記第七章之?dāng)?shù)據(jù)過(guò)濾
這篇文章主要介紹了MYSQL必知必會(huì)讀書筆記第七章之?dāng)?shù)據(jù)過(guò)濾的相關(guān)資料,需要的朋友可以參考下2016-05-05
mysql如何將sql查詢的結(jié)果以百分比展示出來(lái)
這篇文章主要給大家介紹了關(guān)于mysql如何將sql查詢的結(jié)果以百分比展示出來(lái)的相關(guān)資料,用到了MySQL字符串處理中的兩個(gè)函數(shù)concat()和left()實(shí)現(xiàn)查詢結(jié)果以百分比顯示,需要的朋友可以參考下2023-08-08
使用MySQL實(shí)現(xiàn)高效的用戶昵稱模糊搜索
在大型系統(tǒng)中,用戶表中的昵稱字段需要支持高效的模糊搜索,并且必須處理包含特殊字符的查詢,本文將介紹一種在MySQL中實(shí)現(xiàn)高效模糊搜索的解決方案,能夠支持特殊字符,并且利用MySQL自身的全文索引機(jī)制來(lái)優(yōu)化搜索性能,需要的朋友可以參考下2024-05-05
千萬(wàn)級(jí)記錄的Discuz論壇導(dǎo)致MySQL CPU 100%的優(yōu)化筆記
談到自己在解決一個(gè)擁有 60 萬(wàn)條記錄的 MySQL 數(shù)據(jù)庫(kù)訪問(wèn)時(shí),導(dǎo)致 MySQL CPU 占用 100% 的經(jīng)過(guò)。在解決問(wèn)題完成優(yōu)化(optimize)之后,我發(fā)現(xiàn) Discuz 論壇也存在這個(gè)問(wèn)題,當(dāng)時(shí)稍微提了一下2010-12-12

