0

我有一个相当复杂的 POJO,我需要从 JSON 字符串反序列化并保存在 MySQL 数据库中。下面是一个非常简化的示例类:

@Entity
@Table(name="a")
public class A{
    private Long id;
    private B b;
    private C c;
    private D d;
}

@ManyToOne
@JoinColumn(name="b_id")
public B getB(){
    return this.b;
}

public void setB(B b){ this.b = b; }

@ManyToOne
@JoinColumn(name="c_id")
public C getC(){ 
    return this.c;
}

public void setC(C c){ this.c = c; }

@ManyToOne
@JoinColumn(name="d_id")
public D getD(){ 
    return this.d;
}

public void setD(D d){ this.d = d; }

每个类B,CD也有许多字段和对象(有些甚至需要更多的对象和字段)根据数据库模式不能为空,我无法更改。我可以反序列化并坚持这个没问题,但是这样做所需的 JSON 真的很大。我只需要保留反序列化的A,所以我真的只需要_id来自B,CD.

现在我的 JSON 是这样的:

{
  "id":1, 
  "b":{"id":2, ...}, 
  "c":{"id":3, ...}, 
  "d":{"id":4, ...}
}

我必须填写所有不可为空的数据库字段。我想做的是读取一个 JSON 字符串,例如:

{
  "id":1,
  "b_id":2,
  "c_id":3,
  "d_id":4
}

只需让 Hibernate/JPA 更新数据库中的这些字段。我认为真正棘手的部分是我的应用程序中的其他类/方法将需要整个对象层次结构才能从数据库中读取。我只能使用_id字段的唯一时间是在 JSON 的反序列化期间。在这种情况下,我只需要更新最顶层的对象(A在我的示例中)和其他一些琐碎的字段。在一个完美的世界里,我可以在我的课堂上添加一些注释来解决这个问题,但我还没有找到任何能够做到这一点的东西。

这甚至可能吗?如果是这样,可以使用 JPA/Jackson 注释来完成吗?

谢谢!

4

2 回答 2

1

如果将 JSON 直接映射到您的实体是不正确的,我会简单地通过中间对象将传入的 JSON 字符串转换为您的实体(如果您愿意,可以将其称为 DTO)。

我也有一种直觉,如果你保存关联的对象引用(@ManyToOnes)只填充了 id,Hibernate 实际上会正确保存关联(这里的根对象代表 A):

{
    "id" : 1,
    "b" : {
        "id" : 2
    },
    ...
}

确保不要级联 b、c 等字段的持久化操作。

于 2013-06-05T21:18:48.203 回答
0

您应该创建仅定义要反序列化的字段的新类。例如,如果您只想反序列化ID您的类,则可能如下所示:

class JsonEntity {

    private Long id;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    @JsonAnySetter
    public void setUnknownProperties(String name, String value) {
        //do nothing
    }

    @Override
    public String toString() {
        return String.valueOf(id);
    }
}

在此示例中,注释JsonAnySetter可以解决问题。现在,您必须创建一些模拟您的课程的A课程。例如:

class DtoA extends JsonEntity {

    private JsonEntity b;
    private JsonEntity c;
    private JsonEntity d;

    public JsonEntity getB() {
        return b;
    }

    public void setB(JsonEntity b) {
        this.b = b;
    }

    public JsonEntity getC() {
        return c;
    }

    public void setC(JsonEntity c) {
        this.c = c;
    }

    public JsonEntity getD() {
        return d;
    }

    public void setD(JsonEntity d) {
        this.d = d;
    }

    @Override
    public String toString() {
        return "A [id=" + getId() + ", b=" + b + ", c=" + c + ", d=" + d + "]";
    }
}

现在,当我们有新的 JSON 数据模型时,我们可以对其进行测试。例如我们可以解析下面的 JSON:

{
  "id":1, 
  "b":{"id":2, "p1":"v1", "p2":"v2"}, 
  "c":{"id":3, "p3":"v3", "p4":"v4", "p5":"v5"}, 
  "d":{"id":4, "p6":"v6"}
}

反序列化示例:

ObjectMapper objectMapper = new ObjectMapper();
DtoA a = objectMapper.readValue(json, DtoA.class);
System.out.println(a);

上面的程序打印:

A [id=1, b=2, c=3, d=4]

现在,您必须在类中实现toA()方法,DtoA如下所示:

public A toA() {
    A a = new A(getId());
    a.setB(new B(getB().getId()));
    a.setC(new C(getC().getId()));
    a.setD(new D(getD().getId()));

    return a;
}

让我知道它是否适合你。

于 2013-06-05T22:18:08.680 回答