Spring-MVC, Hibernate: создание DTO-объектов из объектов Domain
Я работаю над приложением Spring-MVC, в котором я пытаюсь найти список GroupNotes в базе данных. В моем проекте сопоставление состоит в том, что GroupCanvas имеет сопоставление один ко многим с GroupSection, а GroupSection - сопоставление один ко многим с GroupNotes. Из-за этих сопоставлений я получал LazyInitializationException. Как и предполагалось в SO, я должен преобразовывать объекты в объекты DTO для передачи. Я проверил в сети, но не смог найти подходящий способ перевести их.
Я только что создал новый список, чтобы избежать ошибки, но одно поле все еще дает мне ошибку. Буду признателен, если кто-нибудь скажет мне, как исправить эту ошибку или преобразовать объекты в объекты DTO, чтобы их можно было перенести.
Код контроллера:
@RequestMapping(value = "/findgroupnotes/{days}/{canvasid}")
public @ResponseBody List<GroupNotes> findGroupNotesByDays(@PathVariable("days")int days, @PathVariable("canvasid")int canvasid){
List<GroupNotes> groupNotesList = this.groupNotesService.findGroupNotesByDays(days,canvasid);
List<GroupNotes> toSendList = new ArrayList<>();
for(GroupNotes groupNotes : groupNotesList){
GroupNotes toSendNotes = new GroupNotes();
toSendNotes.setMnotecolor(groupNotes.getMnotecolor());
toSendNotes.setNoteCreationTime(groupNotes.getNoteCreationTime());
toSendNotes.setMnotetag(groupNotes.getMnotetag());
toSendNotes.setMnotetext(groupNotes.getMnotetext());
toSendNotes.setAttachCount(groupNotes.getAttachCount());
toSendNotes.setNoteDate(groupNotes.getNoteDate());
toSendList.add(toSendNotes);
}
return toSendList;
}
GroupNotesDAOImpl:
@Override
public List<GroupNotes> searchNotesByDays(int days, int mcanvasid) {
Session session = this.sessionFactory.getCurrentSession();
Calendar cal = Calendar.getInstance();
cal.add(Calendar.DAY_OF_YEAR, -days);
long daysAgo = cal.getTimeInMillis();
Timestamp nowMinusDaysAsTimestamp = new Timestamp(daysAgo);
GroupCanvas groupCanvas = (GroupCanvas) session.get(GroupCanvas.class,mcanvasid);
Query query = session.createQuery("from GroupSection as n where n.currentcanvas.mcanvasid=:mcanvasid");
query.setParameter("mcanvasid", mcanvasid);
List<GroupSection> sectionList = query.list();
List<GroupNotes> notesList = new ArrayList<GroupNotes>();
for (GroupSection e : sectionList) {
System.out.println("Section name is "+e.getMsectionname());
GroupSection groupSection = (GroupSection) session.get(GroupSection.class,e.getMsectionid());
Query query1 = session.createQuery("from GroupNotes as gn where gn.ownednotes.msectionid=:msectionid and gn.noteCreationTime >:limit");
query1.setParameter("limit", nowMinusDaysAsTimestamp);
query1.setParameter("msectionid",e.getMsectionid());
notesList.addAll(query1.list());
}
// I am getting the data below, but I get JSON errors.
for(GroupNotes groupNotes : notesList){
System.out.println("Group notes found are "+groupNotes.getMnotetext());
}
return notesList;
}
Модель GroupCanvas:
@Entity
@Table(name = "membercanvas")
public class GroupCanvas{
@OneToMany(mappedBy = "currentcanvas",fetch=FetchType.LAZY, cascade = CascadeType.REMOVE)
@JsonIgnore
private Set<GroupSection> ownedsection = new HashSet<>();
@JsonIgnore
public Set<GroupSection> getOwnedsection() {
return this.ownedsection;
}
public void setOwnedsection(Set<GroupSection> ownedsection) {
this.ownedsection = ownedsection;
}
}
Модель GroupSection:
@Entity
@Table(name = "membersection")
public class GroupSection{
@OneToMany(mappedBy = "ownednotes", fetch = FetchType.EAGER,cascade = CascadeType.REMOVE)
@JsonIgnore
private Set<GroupNotes> sectionsnotes = new HashSet<>();
public Set<GroupNotes> getSectionsnotes(){
return this.sectionsnotes;
}
public void setSectionsnotes(Set<GroupNotes> sectionsnotes){
this.sectionsnotes=sectionsnotes;
}
}
Модель GroupNotes:
@Entity
@Table(name="groupnotes")
public class GroupNotes{
@Id
@Column(name="mnoteid")
@GeneratedValue(strategy = GenerationType.SEQUENCE,generator = "mnote_gen")
@SequenceGenerator(name = "mnote_gen",sequenceName = "mnote_seq")
@org.hibernate.annotations.Index(name = "mnoticesidindex")
private int mnoticesid;
@Column(name = "mnotetext")
private String mnotetext;
@Column(name = "mnoteheadline")
private String mnotetag;
@Column(name = "mnotecolor")
private String mnotecolor;
@Column(name = "mnoteorder")
private double mnoteorder;
@Column(name = "attachmentcount")
private int attachCount;
@Column(name = "notedate")
private String noteDate;
@Column(name = "uploader")
private String uploader;
@Column(name = "activeedit")
private boolean activeEdit;
@Column(name = "notedisabled")
private boolean noteDisabled;
@Column(name = "noteinactive")
private boolean noteInActive;
@Column(name = "notecreatoremail")
private String noteCreatorEmail;
@Column(name = "prefix")
private String prefix;
@Column(name = "timestamp")
private Timestamp noteCreationTime;
@Transient
private boolean notRead;
@Transient
private String tempNote;
@Transient
private String canvasUrl;
@ManyToOne
@JoinColumn(name = "msectionid")
@JsonIgnore
private GroupSection ownednotes;
@JsonIgnore
public GroupSection getOwnednotes(){return this.ownednotes;}
public void setOwnednotes(GroupSection ownednotes){this.ownednotes=ownednotes;}
@JsonIgnore
public int getOwnedSectionId(){
return this.ownednotes.getMsectionid();
}
@OneToMany(mappedBy = "mnotedata",fetch = FetchType.LAZY,cascade = CascadeType.REMOVE)
@JsonIgnore
private Set<GroupAttachments> mattachments = new HashSet<>();
public Set<GroupAttachments> getMattachments() {
return this.mattachments;
}
public void setMattachments(Set<GroupAttachments> mattachments) {
this.mattachments = mattachments;
}
@OneToMany(mappedBy = "mhistory",fetch = FetchType.LAZY,cascade = CascadeType.REMOVE)
@JsonIgnore
private Set<GroupNoteHistory> groupNoteHistorySet = new HashSet<>();
public Set<GroupNoteHistory> getGroupNoteHistorySet(){
return this.groupNoteHistorySet;
}
public void setGroupNoteHistorySet(Set<GroupNoteHistory> groupNoteHistorySet){
this.groupNoteHistorySet = groupNoteHistorySet;
}
@OneToMany(mappedBy = "unreadNotes",fetch = FetchType.LAZY,cascade = CascadeType.REMOVE)
@JsonIgnore
private Set<UnreadNotes> unreadNotesSet = new HashSet<>();
public Set<UnreadNotes> getUnreadNotesSet(){
return this.unreadNotesSet;
}
public void setUnreadNotesSet(Set<UnreadNotes> unreadNotesSet){
this.unreadNotesSet = unreadNotesSet;
}
//getters and setters ignored
}
Журнал ошибок:
org.springframework.http.converter.HttpMessageNotWritableException: Could not write JSON: (was java.lang.NullPointerException) (through reference chain: java.util.ArrayList[0]->com.journaldev.spring.model.GroupNotes["ownedSectionId"]); nested exception is com.fasterxml.jackson.databind.JsonMappingException: (was java.lang.NullPointerException) (through reference chain: java.util.ArrayList[0]->com.journaldev.spring.model.GroupNotes["ownedSectionId"])
Пожалуйста, дайте мне знать, что делать, так как я застрял на этой ошибке с некоторого времени.
3 ответа
Я думаю, что это происходит, когда Джексон пытается сериализовать все поля в иерархии на основе методов получения. В какой-то ситуации NullPointerException
бросается в следующем методе:
@JsonIgnore
public int getOwnedSectionId(){
return this.ownednotes.getMsectionid();
}
замените его следующим методом:
@JsonIgnore
public int getOwnedSectionId(){
if(this.ownednotes != null)
return this.ownednotes.getMsectionid();
return 1;
}
У меня нет объяснения, почему Джексон пытается сериализовать его, когда на рынке @JsonIgnore, но вы можете попробовать с моим предложением
Буду признателен, если кто-нибудь скажет мне, как исправить эту ошибку или преобразовать объекты в объекты DTO, чтобы их можно было перенести.
Мы используем DozerMapper на работе для этой цели.
Вместо того, чтобы делать это отображение вручную, вы можете взглянуть на Blaze-Persistence Entity Views, которые можно использовать для эффективной реализации шаблона DTO с JPA. Вот быстрый пример кода, как ваша проблема может быть решена
Сначала вы определяете свой DTO как вид сущности
@EntityView(GroupNotes.class)
public interface GroupNoteView {
@IdMapping("mnoticesid") int getId();
String getMnotecolor();
String getMnotetag();
String getMnotetext();
String getNoteDate();
Timestamp getNoteCreationTime();
int getAttachCount();
}
Затем вы адаптируете свой DAO, чтобы использовать его
@Override
public List<GroupNoteView> searchNotesByDays(int days, int mcanvasid) {
EntityManager entityManager = // get the entity manager from somewhere
CriteriaBuilderFactory cbf = // factory for query building from Blaze-Persistence
EntityViewManager evm = // factory for applying entity views on query builders
Calendar cal = Calendar.getInstance();
cal.add(Calendar.DAY_OF_YEAR, -days);
long daysAgo = cal.getTimeInMillis();
Timestamp nowMinusDaysAsTimestamp = new Timestamp(daysAgo);
CriteriaBuilder<GroupNotes> cb = cbf.create(entityManager, GroupNotes.class, "note")
.where("noteCreationTime").gt(nowMinusDaysAsTimestamp)
.where("ownednotes.ownedcanvas.mcanvasid").eq(mcanvasid);
return evm.applySetting(EntityViewSetting.create(GroupNoteView.class), cb)
.getResultList();
}
И, наконец, код вызова
@RequestMapping(value = "/findgroupnotes/{days}/{canvasid}")
public @ResponseBody List<GroupNoteView> findGroupNotesByDays(@PathVariable("days")int days, @PathVariable("canvasid")int canvasid){
return this.groupNotesService.findGroupNotesByDays(days, canvasid);
}