亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

多層嵌套的json的值如何解析/替換

 更新時間:2023年10月31日 10:49:24   作者:linyb極客之路  
這篇文章主要介紹了多層嵌套的json的值如何解析/替換的方法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

前言

前陣子承接了2個需求,一個數(shù)據(jù)脫敏,一個是低代碼國際化多語言需求,這兩個需求有個共同特點(diǎn),都是以json形式返回給前端,而且都存在多層嵌套,其中數(shù)據(jù)脫敏的數(shù)據(jù)格式是比較固定,而低代碼json的格式存在結(jié)構(gòu)固定和不固定 2種格式。最后不管是數(shù)據(jù)脫敏或者是多語言,業(yè)務(wù)抽象后,都存在需要做json值替換的需求。今天就來聊下多層嵌套json值如何解析或者替換

多層嵌套json解析

1、方法一:循環(huán)遍歷+利用正則進(jìn)行解析

這種做法相對常規(guī),且解析比較繁瑣。

2、方法二:利用OGNL表達(dá)式

1、何為OGNL

OGNL(Object-Graph Navigation Language)是一種表達(dá)式語言,用于在Java應(yīng)用程序中對對象圖進(jìn)行導(dǎo)航和操作。OGNL本身并不提供直接的執(zhí)行環(huán)境,它是作為一個庫或框架的一部分來使用的。因此,OGNL的執(zhí)行方式取決于使用它的上下文。

一般情況下,OGNL可以通過兩種方式執(zhí)行:解釋執(zhí)行和編譯執(zhí)行。

解釋執(zhí)行:在解釋執(zhí)行中,OGNL表達(dá)式在運(yùn)行時逐條解釋和執(zhí)行。它會在每次表達(dá)式執(zhí)行時動態(tài)計(jì)算表達(dá)式的結(jié)果,并根據(jù)對象圖的實(shí)際狀態(tài)進(jìn)行導(dǎo)航和操作。這種方式的靈活性較高,可以根據(jù)需要對對象圖進(jìn)行動態(tài)操作,但相對而言執(zhí)行效率較低。

編譯執(zhí)行:為了提高執(zhí)行效率,有些框架會將OGNL表達(dá)式編譯成可執(zhí)行的字節(jié)碼或類文件。在編譯執(zhí)行中,OGNL表達(dá)式在編譯階段被轉(zhuǎn)換成可執(zhí)行代碼,然后在運(yùn)行時直接執(zhí)行這些生成的代碼。這種方式可以在一定程度上提高執(zhí)行速度,但犧牲了一些靈活性,因?yàn)榫幾g后的代碼在運(yùn)行時不再動態(tài)計(jì)算。

我們經(jīng)常使用ORM框架mybatis的動態(tài)sql解析,它的實(shí)現(xiàn)基石就是OGNL表達(dá)式?;氐秸},我們?nèi)绾卫肙GNL來解析json

a、 在項(xiàng)目POM引入OGNL GAV

<dependency>
            <groupId>ognl</groupId>
            <artifactId>ognl</artifactId>
            <version>${ognl.version}</version>
        </dependency>

b、 封裝OGNL表達(dá)式工具類

public final class OgnlCache {

  private static final OgnlMemberAccess MEMBER_ACCESS = new OgnlMemberAccess();
  private static final OgnlClassResolver CLASS_RESOLVER = new OgnlClassResolver();
  private static final Map<String, Object> expressionCache = new ConcurrentHashMap<>();

  private OgnlCache() {
    // Prevent Instantiation of Static Class
  }

  public static Object getValue(String expression, Object root) {
    try {
      Map context = Ognl.createDefaultContext(root, MEMBER_ACCESS, CLASS_RESOLVER, null);
      return Ognl.getValue(parseExpression(expression), context, root);
    } catch (OgnlException e) {
      throw new RuntimeException("Error evaluating expression '" + expression + "'. Cause: " + e, e);
    }
  }

  private static Object parseExpression(String expression) throws OgnlException {
    Object node = expressionCache.get(expression);
    if (node == null) {
      node = Ognl.parseExpression(expression);
      expressionCache.put(expression, node);
    }
    return node;
  }

}

c、 封裝json工具類

public final class JsonUtil {

    private JsonUtil(){}

    public static <T> T parse(String jsonStr, Class<T> clazz) throws Exception {
        return JSON.parseObject(jsonStr, clazz);
    }

    public static Object getValue(Map map, String path) throws Exception {
         return OgnlCache.getValue(path,map);
    }
}

d、 多層嵌套json解析例子

private void printMenuI18nCodeByOgnl() throws Exception {
        String menuJson = mockMenuService.getMenuJson();
        Map<String, Object> map = JsonUtil.parse(menuJson, Map.class);
        Object topMenu = JsonUtil.getValue( map,"i18NCode");
        Object userMenu = JsonUtil.getValue( map,"children[0].i18NCode");
        Object userMenuAdd = JsonUtil.getValue( map,"children[0].children[0].i18NCode");
        Object userMenuUpdate = JsonUtil.getValue( map,"children[0].children[1].i18NCode");
        Object deptMenu = JsonUtil.getValue( map,"children[1].i18NCode");
        Object deptMenuList = JsonUtil.getValue( map,"children[1].children[0].i18NCode");
        Object deptMenuDelete = JsonUtil.getValue( map,"children[1].children[1].i18NCode");
        System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> Print MenuI18nCode By Ognl Start <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<,");
        System.out.println(topMenu);
        System.out.println(userMenu);
        System.out.println(userMenuAdd);
        System.out.println(userMenuUpdate);
        System.out.println(deptMenu);
        System.out.println(deptMenuList);
        System.out.println(deptMenuDelete);
        System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> Print MenuI18nCode By Ognl End <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<,");
    }

注: 示例中的menuJson形如下

{"children":[{"children":[{"children":[],"component":"saas/index","i18NCode":"user.menu.add","id":8,"linkUrl":"/user/add","menuName":"用戶新增","parentId":9,"sort":9999},{"children":[],"component":"saas/index","i18NCode":"user.menu.update","id":7,"linkUrl":"/user/update","menuName":"用戶編輯","parentId":9,"sort":9999}],"component":"saas/index","i18NCode":"user.menu","id":9,"linkUrl":"/user","menuName":"用戶菜單","parentId":1,"sort":9999},{"children":[{"children":[],"component":"saas/index","i18NCode":"dept.menu.list","id":11,"linkUrl":"/dept/list","menuName":"部門列表","parentId":10,"sort":9999},{"children":[],"component":"saas/index","i18NCode":"dept.menu.delete","id":12,"linkUrl":"/dept/delete","menuName":"部門刪除","parentId":10,"sort":9999}],"component":"saas/index","i18NCode":"dept.menu","id":10,"linkUrl":"/dept","menuName":"部門菜單","parentId":1,"sort":9999}],"component":"saas/index","i18NCode":"top.menu","id":1,"linkUrl":"/topUrl","menuName":"頂級菜單","parentId":0,"sort":9999}

解析后控制臺打印如下

>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> Print MenuI18nCode By Ognl Start <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<,
top.menu
user.menu
user.menu.add
user.menu.update
dept.menu
dept.menu.list
dept.menu.delete
>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> Print MenuI18nCode By Ognl End <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<,

OGNL表達(dá)式常用例子

1. 基本對象樹的訪問

對象樹的訪問就是通過使用點(diǎn)號將對象的引用串聯(lián)起來進(jìn)行。

例如:name,department.name,user.department.factory.manager.name

2. 對容器變量的訪問

對容器變量的訪問,通過#符號加上表達(dá)式進(jìn)行。

例如:#name,#department.name,#user.department.factory.manager.name

3. 使用操作符號

OGNL表達(dá)式中能使用的操作符基本跟Java里的操作符一樣,除了能使用 +, -, *, /, ++, --, ==, !=, = 等操作符之外,還能使用 mod, in, not in等。

4. 容器、數(shù)組、對象

OGNL支持對數(shù)組和ArrayList等容器的順序訪問:

例如:group.users[0]

同時,OGNL支持對Map的按鍵值查找:

例如:#session['mySessionPropKey']

不僅如此,OGNL還支持容器的構(gòu)造的表達(dá)式:

例如:{"green", "red", "blue"}構(gòu)造一個List,#{"key1" : "value1", "key2" : "value2", "key3" : "value3"}構(gòu)造一個Map

你也可以通過任意類對象的構(gòu)造函數(shù)進(jìn)行對象新建:

例如:new java.net.URL("http://localhost/")

5. 對靜態(tài)方法或變量的訪問

要引用類的靜態(tài)方法和字段,他們的表達(dá)方式是一樣的@class@member或者@class@method(args):

例如:@com.javaeye.core.Resource@ENABLE,@com.javaeye.core.Resource@getAllResources

6. 方法調(diào)用

直接通過類似Java的方法調(diào)用方式進(jìn)行,你甚至可以傳遞參數(shù):

例如:user.getName(),group.users.size(),group.containsUser(#requestUser)

7. 投影和選擇

OGNL支持類似數(shù)據(jù)庫中的投影(projection) 和選擇(selection)。

投影就是選出集合中每個元素的相同屬性組成新的集合,類似于關(guān)系數(shù)據(jù)庫的字段操作。投影操作語法為 collection.{XXX},其中XXX 是這個集合中每個元素的公共屬性。

例如:group.userList.{username}將獲得某個group中的所有user的name的列表。

選擇就是過濾滿足selection 條件的集合元素,類似于關(guān)系數(shù)據(jù)庫的紀(jì)錄操作。選擇操作的語法為:collection.{X YYY},其中X 是一個選擇操作符,后面則是選擇用的邏輯表達(dá)式。而選擇操作符有三種:

? 選擇滿足條件的所有元素
^ 選擇滿足條件的第一個元素
$ 選擇滿足條件的最后一個元素

例如:group.userList.{? #this.name != null}將獲得某個group中user的name不為空的user的列表。

多層嵌套json替換

1、方法一:循環(huán)遍歷+正則進(jìn)行替換

這種做法相對常規(guī),且替換比較繁瑣。

2、方法二:利用json類庫,進(jìn)行替換

以fastJSON為例

a、 在項(xiàng)目pom引入fastJSON GAV

<dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>fastjson</artifactId>
            <version>${fastjson.version}</version>
        </dependency>

b、 多層嵌套json替換例子

以將菜單的i18nCode替換為具體語言的值為例

public String reBuildMenuJson(){
        String orginalMenuJson = getMenuJson();
        JSONObject jsonObject = JSON.parseObject(orginalMenuJson);
        jsonObject.put(I18N_CODE_COLUMN,mockI18nCache.get(jsonObject.get(I18N_CODE_COLUMN)));
        reBuildChildJson(jsonObject);
        return JSON.toJSONString(jsonObject);

    }

    private void reBuildChildJson(JSONObject curentObject){
        JSONArray children = curentObject.getJSONArray(CHILDREN_COLUMN);
        for (int i = 0; i < children.size(); i++) {
            JSONObject child = children.getJSONObject(i);
            child.put(I18N_CODE_COLUMN,mockI18nCache.get(child.get(I18N_CODE_COLUMN)));
            reBuildChildJson(child);
        }

    }

注: 未替換前,menuJson形如下

{"children":[{"children":[{"children":[],"component":"saas/index","i18NCode":"user.menu.add","id":8,"linkUrl":"/user/add","menuName":"用戶新增","parentId":9,"sort":9999},{"children":[],"component":"saas/index","i18NCode":"user.menu.update","id":7,"linkUrl":"/user/update","menuName":"用戶編輯","parentId":9,"sort":9999}],"component":"saas/index","i18NCode":"user.menu","id":9,"linkUrl":"/user","menuName":"用戶菜單","parentId":1,"sort":9999},{"children":[{"children":[],"component":"saas/index","i18NCode":"dept.menu.list","id":11,"linkUrl":"/dept/list","menuName":"部門列表","parentId":10,"sort":9999},{"children":[],"component":"saas/index","i18NCode":"dept.menu.delete","id":12,"linkUrl":"/dept/delete","menuName":"部門刪除","parentId":10,"sort":9999}],"component":"saas/index","i18NCode":"dept.menu","id":10,"linkUrl":"/dept","menuName":"部門菜單","parentId":1,"sort":9999}],"component":"saas/index","i18NCode":"top.menu","id":1,"linkUrl":"/topUrl","menuName":"頂級菜單","parentId":0,"sort":9999}

替換后,menuJson形如下

{"component":"saas/index","children":[{"component":"saas/index","children":[{"component":"saas/index","children":[],"linkUrl":"/user/add","menuName":"用戶新增","id":8,"sort":9999,"i18NCode":"userMenuAdd","parentId":9},{"component":"saas/index","children":[],"linkUrl":"/user/update","menuName":"用戶編輯","id":7,"sort":9999,"i18NCode":"userUpdateAdd","parentId":9}],"linkUrl":"/user","menuName":"用戶菜單","id":9,"sort":9999,"i18NCode":"userMenu","parentId":1},{"component":"saas/index","children":[{"component":"saas/index","children":[],"linkUrl":"/dept/list","menuName":"部門列表","id":11,"sort":9999,"i18NCode":"deptMenuList","parentId":10},{"component":"saas/index","children":[],"linkUrl":"/dept/delete","menuName":"部門刪除","id":12,"sort":9999,"i18NCode":"deptMenuDelete","parentId":10}],"linkUrl":"/dept","menuName":"部門菜單","id":10,"sort":9999,"i18NCode":"deptMenu","parentId":1}],"linkUrl":"/topUrl","menuName":"頂級菜單","id":1,"sort":9999,"i18NCode":"topMenu","parentId":0}

3、方法三:利用json序列化注解

以菜單國際化為示例

1、自定義注解

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@JacksonAnnotationsInside
@JsonSerialize(using = I18nJsonSerializer.class)
public @interface I18nField {
}

2、自定義國際化翻譯接口(該具體實(shí)現(xiàn)留給業(yè)務(wù)擴(kuò)展)

public interface I18nService {

    String getTargetContent(String i18nCode);
}

題外話 : 為啥不像spring的messageSource定義成

String getMessage(String code, @Nullable Object[] args, @Nullable String defaultMessage, Locale locale);

因?yàn)楹芏鄥?shù)信息可以直接通過上下文獲取,比如Locale可以通過LocaleContextHolder.getLocale()

3、編寫json序列化接口

public class I18nJsonSerializer extends JsonSerializer<String> implements ContextualSerializer {
    @Autowired
    private I18nService i18nService;
    @Override
    public void serialize(String s, JsonGenerator jsonGenerator, SerializerProvider serializerProvider) throws IOException {
        jsonGenerator.writeString(i18nService.getTargetContent(s));
    }
    @Override
    public JsonSerializer<?> createContextual(SerializerProvider serializerProvider, BeanProperty beanProperty) throws JsonMappingException {
        I18nField i18nField = beanProperty.getAnnotation(I18nField.class);
        if(!ObjectUtils.isEmpty(i18nField) && String.class.isAssignableFrom(beanProperty.getType().getRawClass())){
            return this;
        }
        return serializerProvider.findValueSerializer(beanProperty.getType(),beanProperty);
    }
}

4、定義和json字段能夠匹配的對象

大白話,就是json和這個對象可以相互轉(zhuǎn)換。以菜單為例

@Data
@EqualsAndHashCode(callSuper = true, of = {"id"})
public class MenuResourceDTO extends TreeDTO<MenuResourceDTO> implements Serializable {
    private static final long serialVersionUID = 1L;
 。。。。。省略其他屬性
    /**
     * 單菜名稱
     */
    private String menuName;
    private String permission;
    /**
     * 是否緩存
     */
    private Integer keepAlive;
    @I18nField
    private String i18NCode;
    public static String I18N_CODE_COLUMN = "i18NCode";
    public static String CHILDREN_COLUMN = "children";

5、在需要進(jìn)行替換的字段上加上 @I18nField注解

@I18nField
    private String i18NCode;

6、替換驗(yàn)證

編寫一個測試controller,用來輸出替換后的菜單信息

@RestController
@RequestMapping("menu")
@RequiredArgsConstructor
public class MockMenuController {
    private final MockMenuService mockMenuService;
    @GetMapping
    public MenuResourceDTO getMenu(){
        return mockMenuService.getMenuResourceDTO();
    }
}

通過POSTMAN訪問,得到如下信息

{
    "id": 1,
    "parentId": 0,
    "sort": 9999,
    "children": [
        {
            "id": 9,
            "parentId": 1,
            "sort": 9999,
            "children": [
                {
                    。。。省略其他信息
                    "menuName": "用戶新增",
                    "i18NCode": "userMenuAdd"
                },
                {
                    。。。省略其他信息
                    "menuName": "用戶編輯",
                    "i18NCode": "userUpdateAdd"
                }
            ],
            。。。省略其他信息
            "menuName": "用戶菜單",
            "i18NCode": "userMenu"
        },
    "menuName": "頂級菜單",
    "i18NCode": "topMenu"
}

回答上面多層json解析的方法三,那個懸念做法就是將json與對象映射起來,通過對象來取值

總結(jié)

本文的多層嵌套json的解析和替換都提供了幾種方案,綜合來講是推薦將json先轉(zhuǎn)對象,通過對象操作。對json替換,推薦使用自定義json序列化注解的方式。但這種方式比較適合json的結(jié)構(gòu)以及字段是固定的方式。對于低代碼,本身的json結(jié)構(gòu)是多種多樣的,如果要后端實(shí)現(xiàn),一種做法,就是將這些json都映射成對象,但因?yàn)閖son結(jié)構(gòu)多種多樣,就會導(dǎo)致要映射的對象膨脹。另一種方式,是直接轉(zhuǎn)JsonObject,通過JsonObject來操作替換

其次現(xiàn)在都是前后端分離,有些東西其實(shí)也可以放在前端實(shí)現(xiàn),比如這種替換工作其實(shí)挺適合放在前端做的。以低代碼為例,因?yàn)榍岸吮緛砭托枰馕鰆son,后端可以維護(hù)一個映射表,前端實(shí)現(xiàn)一個組件函數(shù),通過該函數(shù)優(yōu)先從前端緩存取,取不到再從調(diào)用后端接口,這就是json替換的方法四,把替換工作留給前端做,哈哈。大家是一個團(tuán)隊(duì),哪邊好實(shí)現(xiàn),就放哪邊做

最后那個ognl的代碼,我是直接把mybatis的源碼搬過來,直接套用了。開源有的東西,就沒必要自己再搞一遍了

demo鏈接

以上就是多層嵌套的json的值如何解析/替換的詳細(xì)內(nèi)容,更多關(guān)于多層嵌套json值解析替換的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Flutter實(shí)現(xiàn)文本組件、圖標(biāo)及按鈕組件的代碼

    Flutter實(shí)現(xiàn)文本組件、圖標(biāo)及按鈕組件的代碼

    這篇文章主要介紹了Flutter實(shí)現(xiàn)文本組件、圖標(biāo)及按鈕組件的代碼,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價值 ,需要的朋友可以參考下
    2019-07-07
  • Springboot整合分頁插件PageHelper步驟解析

    Springboot整合分頁插件PageHelper步驟解析

    這篇文章主要介紹了Springboot整合分頁插件PageHelper步驟解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-06-06
  • Spring Boot全局異常處理解析

    Spring Boot全局異常處理解析

    這篇文章主要為大家詳細(xì)介紹了Spring Boot全局異常處理的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • java實(shí)習(xí)--每天打卡十道面試題!

    java實(shí)習(xí)--每天打卡十道面試題!

    臨近秋招,備戰(zhàn)暑期實(shí)習(xí),祝大家每天進(jìn)步億點(diǎn)點(diǎn)!本篇文章準(zhǔn)備了十道java的常用面試題,希望能夠給大家提供幫助,最后祝大家面試成功,進(jìn)入自己心儀的大廠
    2021-07-07
  • 淺談Java操作符與其優(yōu)先級

    淺談Java操作符與其優(yōu)先級

    這篇文章主要介紹了淺談Java操作符與其優(yōu)先級,具有一定借鑒價值,需要的朋友可以了解下。
    2017-12-12
  • Java中的Hashtable源碼詳細(xì)解析

    Java中的Hashtable源碼詳細(xì)解析

    這篇文章主要介紹了Java中的Hashtable源碼詳細(xì)解析,Hashtable 的函數(shù)都是同步的,這意味著它是線程安全的,它的key、value都不可以為null,此外,Hashtable中的映射不是有序的,需要的朋友可以參考下
    2023-11-11
  • java使用反射訪問成員變量的值示例

    java使用反射訪問成員變量的值示例

    這篇文章主要介紹了java使用反射訪問成員變量的值,結(jié)合實(shí)例形式分析了java基于反射機(jī)制操作類成員變量相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下
    2019-07-07
  • Java反射入門、原理與使用方法詳解

    Java反射入門、原理與使用方法詳解

    這篇文章主要介紹了Java反射入門、原理與使用方法,結(jié)合實(shí)例形式詳細(xì)分析了java反射的概念、原理、使用方法與相關(guān)操作注意事項(xiàng),需要的朋友可以參考下
    2015-07-07
  • 跟我學(xué)Java Swing之游戲設(shè)計(jì)(1)

    跟我學(xué)Java Swing之游戲設(shè)計(jì)(1)

    跟我學(xué)Java Swing之游戲設(shè)計(jì)(1)...
    2006-12-12
  • Spring?Boot應(yīng)用中如何動態(tài)指定數(shù)據(jù)庫實(shí)現(xiàn)不同用戶不同數(shù)據(jù)庫的問題

    Spring?Boot應(yīng)用中如何動態(tài)指定數(shù)據(jù)庫實(shí)現(xiàn)不同用戶不同數(shù)據(jù)庫的問題

    讓我們創(chuàng)建一個 Spring Boot 項(xiàng)目首先設(shè)置一個具有必要依賴項(xiàng)的新 Spring Boot項(xiàng)目,在項(xiàng)目配置中包括 Spring Web、Spring Data JPA 和關(guān)于數(shù)據(jù)庫的依賴項(xiàng),接下來介紹Spring?Boot應(yīng)用中如何動態(tài)指定數(shù)據(jù)庫,實(shí)現(xiàn)不同用戶不同數(shù)據(jù)庫的場景?,需要的朋友可以參考下
    2024-04-04

最新評論