จะแมปค่าที่ซ้อนกันกับคุณสมบัติโดยใช้คำอธิบายประกอบ Jackson ได้อย่างไร


93

สมมติว่าฉันกำลังโทรไปยัง API ที่ตอบสนองด้วย JSON ต่อไปนี้สำหรับผลิตภัณฑ์:

{
  "id": 123,
  "name": "The Best Product",
  "brand": {
     "id": 234,
     "name": "ACME Products"
  }
}

ฉันสามารถแมปรหัสผลิตภัณฑ์และชื่อได้โดยใช้คำอธิบายประกอบของ Jackson:

public class ProductTest {
    private int productId;
    private String productName, brandName;

    @JsonProperty("id")
    public int getProductId() {
        return productId;
    }

    public void setProductId(int productId) {
        this.productId = productId;
    }

    @JsonProperty("name")
    public String getProductName() {
        return productName;
    }

    public void setProductName(String productName) {
        this.productName = productName;
    }

    public String getBrandName() {
        return brandName;
    }

    public void setBrandName(String brandName) {
        this.brandName = brandName;
    }
}

จากนั้นใช้เมธอด fromJson เพื่อสร้างผลิตภัณฑ์:

  JsonNode apiResponse = api.getResponse();
  Product product = Json.fromJson(apiResponse, Product.class);

แต่ตอนนี้ฉันกำลังพยายามหาวิธีคว้าชื่อแบรนด์ซึ่งเป็นคุณสมบัติซ้อนกัน ฉันหวังว่าสิ่งนี้จะได้ผล:

    @JsonProperty("brand.name")
    public String getBrandName() {
        return brandName;
    }

แต่แน่นอนมันไม่ได้ มีวิธีง่ายๆในการทำสิ่งที่ต้องการโดยใช้คำอธิบายประกอบหรือไม่

การตอบสนอง JSON จริงที่ฉันพยายามแยกวิเคราะห์นั้นซับซ้อนมากและฉันไม่ต้องการสร้างคลาสใหม่ทั้งหมดสำหรับทุกโหนดย่อยแม้ว่าฉันจะต้องการเพียงช่องเดียวก็ตาม


2
ฉันลงเอยด้วยการใช้github.com/json-path/JsonPath - Spring ใช้มันใต้ฝากระโปรงเช่นกัน ตัวอย่างเช่นใน org.springframework.data.web
dehumanizer

คำตอบ:


91

คุณสามารถบรรลุสิ่งนี้:

String brandName;

@JsonProperty("brand")
private void unpackNameFromNestedObject(Map<String, String> brand) {
    brandName = brand.get("name");
}

24
สามระดับลึกแค่ไหน?
Robin Kanters

5
@ โรบินจะไม่ทำงานเหรอ? this.abbreviation = ((Map<String, Object>)portalCharacteristics.get("icon")).get("ticker").toString();
Marcello de Sales

สามารถใช้วิธีการเดียวกันสำหรับการคลายหลายค่าได้เช่นกัน ... unpackValuesFromNestedBrand - ขอบคุณ
msanjay

13

นี่คือวิธีที่ฉันจัดการปัญหานี้:

Brand ชั้น:

package org.answer.entity;

public class Brand {

    private Long id;

    private String name;

    public Brand() {

    }

    //accessors and mutators
}

Product ชั้น:

package org.answer.entity;

import com.fasterxml.jackson.annotation.JsonGetter;
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.annotation.JsonSetter;

public class Product {

    private Long id;

    private String name;

    @JsonIgnore
    private Brand brand;

    private String brandName;

    public Product(){}

    @JsonGetter("brandName")
    protected String getBrandName() {
        if (brand != null)
            brandName = brand.getName();
        return brandName;
    }

    @JsonSetter("brandName")
    protected void setBrandName(String brandName) {
        if (brandName != null) {
            brand = new Brand();
            brand.setName(brandName);
        }
        this.brandName = brandName;
    }

//other accessors and mutators
}

ที่นี่brandเช่นจะได้รับการปฏิเสธโดยJacksonในระหว่างserializationและเนื่องจากมันถูกกำกับด้วยdeserialization@JsonIgnore

Jacksonจะใช้วิธีการใส่คำอธิบายประกอบด้วย@JsonGetterfor serializationof java object ในJSONรูปแบบ ดังนั้นจึงbrandNameตั้งค่าด้วยbrand.getName().

ในทำนองเดียวกันJacksonจะใช้วิธีการใส่คำอธิบายประกอบด้วย@JsonSetterfor deserializationof JSONformat ลงในวัตถุ java ในสถานการณ์นี้คุณจะต้องยกตัวอย่างbrandวัตถุตัวเองและการตั้งค่าของทรัพย์สินจากnamebrandName

คุณสามารถใช้@Transientคำอธิบายประกอบแบบคงอยู่brandNameได้หากคุณต้องการให้ผู้ให้บริการการคงอยู่ละเว้น


5

คุณสามารถใช้ JsonPath-expression เพื่อแมปคุณสมบัติที่ซ้อนกัน ฉันไม่คิดว่าจะมีการสนับสนุนอย่างเป็นทางการ (ดูปัญหานี้ ) แต่มีการนำไปใช้อย่างไม่เป็นทางการที่นี่: https://github.com/elasticpath/json-unmarshaller


1

วิธีที่ดีที่สุดคือใช้วิธี setter:

JSON:

...
 "coordinates": {
               "lat": 34.018721,
               "lng": -118.489090
             }
...

วิธี setter สำหรับ lat หรือ lng จะมีลักษณะดังนี้:

 @JsonProperty("coordinates")
    public void setLng(Map<String, String> coordinates) {
        this.lng = (Float.parseFloat(coordinates.get("lng")));
 }

หากคุณต้องการอ่านทั้งสองอย่าง (ตามปกติ) ให้ใช้วิธีการที่กำหนดเอง

@JsonProperty("coordinates")
public void setLatLng(Map<String, String> coordinates){
    this.lat = (Float.parseFloat(coordinates.get("lat")));
    this.lng = (Float.parseFloat(coordinates.get("lng")));
}

-6

เพื่อให้ง่าย .. ฉันได้เขียนโค้ด ... ส่วนใหญ่จะอธิบายด้วยตนเอง

Main Method

package com.test;

import java.io.IOException;

import com.fasterxml.jackson.core.JsonParseException;
import com.fasterxml.jackson.databind.JsonMappingException;
import com.fasterxml.jackson.databind.ObjectMapper;

public class LOGIC {

    public static void main(String[] args) throws JsonParseException, JsonMappingException, IOException {

        ObjectMapper objectMapper = new ObjectMapper();
        String DATA = "{\r\n" + 
                "  \"id\": 123,\r\n" + 
                "  \"name\": \"The Best Product\",\r\n" + 
                "  \"brand\": {\r\n" + 
                "     \"id\": 234,\r\n" + 
                "     \"name\": \"ACME Products\"\r\n" + 
                "  }\r\n" + 
                "}";

        ProductTest productTest = objectMapper.readValue(DATA, ProductTest.class);
        System.out.println(productTest.toString());

    }

}

Class ProductTest

package com.test;

import com.fasterxml.jackson.annotation.JsonProperty;

public class ProductTest {

    private int productId;
    private String productName;
    private BrandName brandName;

    @JsonProperty("id")
    public int getProductId() {
        return productId;
    }
    public void setProductId(int productId) {
        this.productId = productId;
    }

    @JsonProperty("name")
    public String getProductName() {
        return productName;
    }
    public void setProductName(String productName) {
        this.productName = productName;
    }

    @JsonProperty("brand")
    public BrandName getBrandName() {
        return brandName;
    }
    public void setBrandName(BrandName brandName) {
        this.brandName = brandName;
    }

    @Override
    public String toString() {
        return "ProductTest [productId=" + productId + ", productName=" + productName + ", brandName=" + brandName
                + "]";
    }



}

Class BrandName

package com.test;

public class BrandName {

    private Integer id;
    private String name;
    public Integer getId() {
        return id;
    }
    public void setId(Integer id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    @Override
    public String toString() {
        return "BrandName [id=" + id + ", name=" + name + "]";
    }




}

OUTPUT

ProductTest [productId=123, productName=The Best Product, brandName=BrandName [id=234, name=ACME Products]]

6
วิธีนี้ใช้ได้ผล แต่ฉันกำลังพยายามหาวิธีแก้ปัญหาโดยที่ฉันไม่ต้องสร้างคลาสใหม่สำหรับทุกโหนดแม้ว่าฉันจะต้องการเพียงฟิลด์เดียวก็ตาม
kenske

-7

สวัสดีนี่คือรหัสการทำงานที่สมบูรณ์

// JUNIT TEST CLASS

สาธารณะชั้นนุ่ม {

@Test
public void test() {

    Brand b = new Brand();
    b.id=1;
    b.name="RIZZE";

    Product p = new Product();
    p.brand=b;
    p.id=12;
    p.name="bigdata";


    //mapper
    ObjectMapper o = new ObjectMapper();
    o.registerSubtypes(Brand.class);
    o.registerSubtypes(Product.class);
    o.setVisibility(PropertyAccessor.FIELD, Visibility.ANY);

    String json=null;
    try {
        json = o.writeValueAsString(p);
        assertTrue(json!=null);
        logger.info(json);

        Product p2;
        try {
            p2 = o.readValue(json, Product.class);
            assertTrue(p2!=null);
            assertTrue(p2.id== p.id);
            assertTrue(p2.name.compareTo(p.name)==0);
            assertTrue(p2.brand.id==p.brand.id);
            logger.info("SUCCESS");
        } catch (IOException e) {

            e.printStackTrace();
            fail(e.toString());
        }




    } catch (JsonProcessingException e) {

        e.printStackTrace();
        fail(e.toString());
    }


}
}




**// Product.class**
    public class Product {
    protected int id;
    protected String name;

    @JsonProperty("brand") //not necessary ... but written
    protected Brand brand;

}

    **//Brand class**
    public class Brand {
    protected int id;
    protected String name;     
}

//Console.log ของ Junit testcase

2016-05-03 15:21:42 396 INFO  {"id":12,"name":"bigdata","brand":{"id":1,"name":"RIZZE"}} / MReloadDB:40 
2016-05-03 15:21:42 397 INFO  SUCCESS / MReloadDB:49 

ข้อมูลสำคัญทั้งหมด: https://gist.github.com/jeorfevre/7c94d4b36a809d4acf2f188f204a8058


1
การตอบสนอง JSON จริงที่ฉันพยายามแมปนั้นซับซ้อนมาก มันมีโหนดและโหนดย่อยจำนวนมากและการสร้างคลาสสำหรับแต่ละโหนดจะเจ็บปวดมากดังนั้นในกรณีส่วนใหญ่ฉันต้องการเพียงฟิลด์เดียวจากชุดของโหนดที่ซ้อนกันสามโหนด ไม่มีวิธีใดที่จะได้รับเขตข้อมูลเดียวโดยไม่ต้องสร้างชั้นเรียนใหม่มากมาย?
kenske

เปิดตั๋วใหม่และแบ่งปันกับฉันฉันสามารถช่วยคุณในเรื่องนี้ แบ่งปัน json struture ที่คุณต้องการแผนที่กรุณา :)
jeorfevre
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.