java.lang.StackOverflowError для методов получения gson - PullRequest
0 голосов
/ 01 марта 2019

Мой класс питомцев выглядит следующим образом (не принимал геттеры и сеттеры).

@Entity
@Table(name = "pet")
@NamedQueries({
    @NamedQuery(name = "Pet.findAll", query = "SELECT p FROM Pet p")
    , @NamedQuery(name = "Pet.findById", query = "SELECT p FROM Pet p WHERE p.id = :id")
    , @NamedQuery(name = "Pet.findByName", query = "SELECT p FROM Pet p WHERE p.name = :name")
    , @NamedQuery(name = "Pet.findByBirth", query = "SELECT p FROM Pet p WHERE p.birth = :birth")
    , @NamedQuery(name = "Pet.findBySpecies", query = "SELECT p FROM Pet p WHERE p.species = :species")
    , @NamedQuery(name = "Pet.findByDeath", query = "SELECT p FROM Pet p WHERE p.death = :death")})
public class Pet implements Serializable {

    private static final long serialVersionUID = 1L;
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Basic(optional = false)
    @Column(name = "id", nullable = false)
    private Integer id;
    @Basic(optional = false)
    @NotNull
    @Size(min = 1, max = 45)
    @Column(name = "name", nullable = false, length = 45)
    private String name;
    @Basic(optional = false)
    @NotNull
    @Column(name = "birth", nullable = false)
    @Temporal(TemporalType.DATE)
    private Date birth;
    @Basic(optional = false)
    @NotNull
    @Size(min = 1, max = 45)
    @Column(name = "species", nullable = false, length = 45)
    private String species;
    @Column(name = "death")
    @Temporal(TemporalType.DATE)
    private Date death;
    @OneToMany(cascade = CascadeType.ALL, mappedBy = "pet")
    private Collection<Event> eventCollection;
    @JoinColumn(name = "owner_id", referencedColumnName = "id")
    @ManyToOne
    private Owner owner;

    public Pet() {
    }

    public Pet(Integer id) {
        this.id = id;
    }

    public Pet(Integer id, String name, Date birth, String species) {
        this.id = id;
        this.name = name;
        this.birth = birth;
        this.species = species;
    }

У меня есть этот метод gson

@GET 
    @Path("/living")
    @Produces(MediaType.APPLICATION_JSON)
    public Response getLivingPets(){
        return Response.ok().entity(gson.toJson(pf.getLivingPets())).build();
    }

и метод фасада моей базы данных

public List<Pet> getLivingPets() {

        EntityManager em = emf.createEntityManager();
        try {
            Query q = em.createQuery("select p from Pet p where p.death is null");

            return (List<Pet>) q.getResultList();
        } finally {
            em.close();
        }
    }

Я получаю эту ошибку

тип Отчет об исключении

сообщение java.lang.StackOverflowError

description Сервер обнаружил внутреннюю ошибку, котораяне позволил ему выполнить этот запрос.

исключение

javax.servlet.ServletException: java.lang.StackOverflowError com.sun.jersey.spi.container.servlet.WebComponent.service (WebComponent.java: 420) com.sun.jersey.spi.container.servlet.ServletContainer.service (ServletContainer.java:558) com.sun.jersey.spi.container.servlet.ServletContainer.service (ServletContainer.java:733) javax.servlet.http.HttpServlet.service (HttpServlet.java:725) org.apache.tomcat.websocket.server.WsFilter.doFilter (WsFilter.java:52) org.netbeans.modules.web.monitor.server.MonitorFilterFilterF.java: 393)

root caиспользуйте

java.lang.StackOverflowError com.google.gson.stream.JsonWriter.beforeValue (JsonWriter.java:642) com.google.gson.stream.JsonWriter.open (JsonWriter.java:325) ком.google.gson.stream.JsonWriter.beginObject (JsonWriter.java:308) com.google.gson.internal.bind.ReflectiveTypeAdapterFactory $ Adapter.write (ReflectiveTypeAdapterFactory.java:240) com.google.gson.Grite $ FutureTyGson.java:1018) com.google.gson.internal.bind.TypeAdapterRuntimeTypeWrapper.write (TypeAdapterRuntimeTypeWrapper.java:69)

1 Ответ

0 голосов
/ 05 марта 2019

первое изменение Если вам нужно закрыть поток, то рекомендуется использовать инструкцию try-with-resources ссылка

    try (EntityManager em = emf.createEntityManager()) {
        Query q = em.createQuery("select p from Pet p where p.death is null");

        return (List<Pet>) q.getResultList();
    }

версия 1 отправлять только d отправлять тольконужное вам поле

@GET 
@Path("/living")
@Produces(MediaType.APPLICATION_JSON)
public Response getLivingPets(){
    return Response.ok().entity(gson.toJson(pf.getLivingPets().stream().map(Dto::new).collect(Collectors.toList()))).build();
}

// adn создал класс вашего DTO

public class Dto {
    private final Integer id;
    private final String name;

    public Dto(Pet pet) {
        this.id= pet.getId();
        this.name= pet.getName();
    }
}

версия 2 create TypeAdapter, где находится ссылка обратной связи для Pet

@GET 
@Path("/living")
@Produces(MediaType.APPLICATION_JSON)
public Response getLivingPets(){
    Gson gson = new GsonBuilder().registerTypeAdapter(Owner.class, new PetAdapter()).create();
    return Response.ok().entity(gson.toJson(pf.getLivingPets())).build();
}

PetAdapter

public class PetAdapter extends TypeAdapter<Owner> {

    @Override
    public void write(JsonWriter out, Owner value) throws IOException {

        out.beginObject();
        out.name("id");
        out.value(value.getId());
        out.endObject();

        //NOT USE PET
    }

    @Override
    public Owner read(JsonReader in) throws IOException {
        return null;
    }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...