java - 使用与 json 相关的 Manytoone 和 onetomany 序列化 list<object>

标签 java json playframework playframework-2.0 jsonserializer

我有类菜单,它是一个 self 与多对多和多对多关系的 self 。

package models;

import java.util.*;
import javax.persistence.*;
import play.db.ebean.*;
import play.data.format.*;
import play.data.validation.*;
import static play.data.validation.Constraints.*;
import javax.validation.*;

import org.codehaus.jackson.annotate.JsonBackReference;
import org.codehaus.jackson.annotate.JsonIgnore;
import org.codehaus.jackson.annotate.JsonManagedReference;

import com.avaje.ebean.*;
import play.i18n.Messages;
@Entity
public class Menu extends Model {

@Id
@GeneratedValue(strategy = GenerationType.AUTO)
public Long id;

@Required
@MinLength(4)
@MaxLength(30)
public String name;

public String url;

@Transient
public boolean hasChild() {
    return url.isEmpty();
}

public Integer idx;

@Temporal(TemporalType.TIMESTAMP)
@Formats.DateTime(pattern = "yyyy/MM/dd hh:mm:ss")
public Date created;

@Required
public boolean enabled;

@ManyToOne
        @JsonBackReference
public Menu parent;

@OneToMany
@JsonManagedReference("parent")
public List<Menu> children;

public static Model.Finder<Long, Menu> find = new Model.Finder<Long, Menu>(Long.class, Menu.class);

public static List<Menu> findTops() {
    return find.where().eq("parent_id", null).eq("enabled", true).orderBy("idx asc").findList();
}


public static List<Menu> findChildsByParent(Menu parent) {
    return findChildsByParent(parent, true);
}

public static List<Menu> findChildsByParent(Menu parent, boolean enabled) {
    return find.where().eq("parent_id", parent.id).eq("enabled", enabled).orderBy("idx asc").findList();
}

public static boolean hasChilds(Menu parent) {
    return hasChilds(parent, true);
}

public static boolean hasChilds(Menu parent, boolean enabled) {
    return find.where().eq("parent_id", parent.id).eq("enabled", enabled).findList().size() > 0;
}

public static Page<Menu> findPage(int page, int size) {
    return find.findPagingList(size).getPage(page - 1);
}


public Menu() {
}
}

在 Controller 中代码是:

@BodyParser.Of(BodyParser.Json.class)
public static Result menuJson() {
    if (menus == null) {
        menus = Menu.find.all();
    }

    JsonNode json = new ObjectMapper().valueToTree(menus);

    return ok(json);
}   

错误详细信息是:

[RuntimeException: java.lang.IllegalArgumentException: Query threw SQLException:Unknown column 't1.menu_id' in 'on clause' Bind values:[1] Query was: select t0.id c0 , t1.id c1, t1.name c2, t1.url c3, t1.idx c4, t1.created c5, t1.enabled c6, t1.parent_id c7 from menu t0 left outer join menu t1 on t1.menu_id = t0.id where t0.id = ? order by t0.id (through reference chain: com.avaje.ebean.common.BeanList[0]->models.Menu["children"])] 

有没有好的解决方案来解决这些问题或者如何声明自定义序列化?对于树模型,我没有很好的类设计对象,对于这个环境是否有更好的设计?

最佳答案

我解决了它们。

在 OneToMany 上添加映射是可行的。

package models;

import java.util.*;
import javax.persistence.*;
import play.db.ebean.*;
import play.data.format.*;
import play.data.validation.*;
import static play.data.validation.Constraints.*;
import javax.validation.*;

import org.codehaus.jackson.annotate.JsonBackReference;
import org.codehaus.jackson.annotate.JsonIgnore;
import org.codehaus.jackson.annotate.JsonManagedReference;

import com.avaje.ebean.*;
import play.i18n.Messages;

@Entity
public class Menu extends Model {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    public Long id;

    @Required
    @MinLength(4)
    @MaxLength(30)
    public String name;

    public String url;

    @Transient
    public boolean hasChild() {
        return url.isEmpty();
    }

    public Integer idx;

    @Temporal(TemporalType.TIMESTAMP)
    @Formats.DateTime(pattern = "yyyy/MM/dd hh:mm:ss")
    public Date created;

    @Required
    public boolean enabled;

    @ManyToOne
    @JsonBackReference
    public Menu parent;

    @OneToMany(mappedBy = "parent", cascade = CascadeType.PERSIST)
    @JsonManagedReference
    public List<Menu> children;

    public static Model.Finder<Long, Menu> find = new Model.Finder<Long, Menu>(Long.class, Menu.class);

    public static List<Menu> findTops() {
        return find.where().eq("parent_id", null).eq("enabled", true).orderBy("idx asc").findList();
    }


    public static List<Menu> findChildsByParent(Menu parent) {
        return findChildsByParent(parent, true);
    }

    public static List<Menu> findChildsByParent(Menu parent, boolean enabled) {
        return find.where().eq("parent_id", parent.id).eq("enabled", enabled).orderBy("idx asc").findList();
    }

    public static boolean hasChilds(Menu parent) {
        return hasChilds(parent, true);
    }

    public static boolean hasChilds(Menu parent, boolean enabled) {
        return find.where().eq("parent_id", parent.id).eq("enabled", enabled).findList().size() > 0;
    }

    public static Page<Menu> findPage(int page, int size) {
        return find.findPagingList(size).getPage(page - 1);
    }


    public Menu() {
    }
}

关于java - 使用与 json 相关的 Manytoone 和 onetomany 序列化 list<object>,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13785530/

相关文章:

java - 在运行时动态禁用 hibernate 验证注释?

java - 使用 firestore 在 for 循环中设置 boolean 值仅适用于列表中的一个对象

java - 使用模数和指数在 C# 中进行 RSA/ECB/PKCS1Padding 解密

java - 在 Play Framework 中创建自定义的 404/500 错误页面

scala - 开源精心设计的 play-scala-slick 应用程序

java - 如何让创建的 Pod 运行应用程序(命令和参数),同时让部署和服务引用它?

c# - Newtonsoft Json 和 System.ObjectDisposedException 与 Xamarin 方法

javascript - 实时流程图不适用于 JSON

php - json_encode 不能转义单引号?

json - 玩 & ReactiveMongo : How to deserialize Json from request in controller