Mam problem z zapytaniem jpa. Istnieją dwie tabele, tj. Tabela Post i tablica tagów Istnieje wiele do wielu relacji między postem a tagiemIN i = operator w języku zapytań JPA
Teraz chcę napisać zapytanie, które po wybraniu wielu tagów, wszystkie posty powiązane z tymi tagami powinny być wybrany. Na przykład
post1 has tags friends and motivation
post2 has tags motivation and pune
post3 has tag boxing
jeśli tagi przyjaciół i Pune są wybierane następnie post1 i po 2 powinny być pobierane jeśli tag boks jest wybierany tylko wtedy zakładać 3 powinny być pobierane jeśli tagi boks i motywacja są wybierane wtedy wszystkie trzy posty powinny zostać odzyskane.
Próbowałem następujące rzeczy
SELECT DISTINCT p FROM Post p JOIN p.tags tags WHERE p.tags IN :tags
ale daje błąd walidatora że
The state field path 'p.tags' cannot be resolved to a collection type.
Jeśli staram się ten
SELECT DISTINCT p FROM Post p JOIN p.tags tags WHERE p.tags = :tags
to spełnia w porządku, ale po przejściu listę tagi daje błąd
java.lang.IllegalArgumentException: You have attempted to set a value of type class java.util.ArrayList for parameter tags with expected type of class com.justme.model.entities.Tag from query string SELECT DISTINCT p FROM Post p JOIN p.tags tags WHERE p.tags = :tags.
Bardzo dziękuję za przeczytanie :) Czy możesz mnie o tym poprowadzić?
Jak mogę osiągnąć powyższe wyniki? mój dostawca wytrwałość jest EclipseLink
Jest to jednostka post
@Entity
@NamedQueries({
@NamedQuery(name = "Post.selectAllPosts", query = "SELECT p FROM Post p ORDER BY p.dateCreated DESC"),
@NamedQuery(name = "Post.selectPostForUser", query = "SELECT p FROM Post p WHERE p.user = :user ORDER BY p.dateCreated DESC"),
@NamedQuery(name = "Post.selectPostsByTags", query = "SELECT DISTINCT p FROM Post p JOIN p.tags tags WHERE p.tags IN :tags") })
public class Post implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private int idpost;
@Lob
private String content;
private String title;
// bi-directional many-to-one association to User
@ManyToOne(cascade = CascadeType.PERSIST)
@JoinColumn(name = "iduser")
private User user;
// bi-directional many-to-many association to Tag
@ManyToMany(cascade = CascadeType.PERSIST)
@JoinTable(name = "post_tag", joinColumns = @JoinColumn(name = "idpost"), inverseJoinColumns = @JoinColumn(name = "idtag"))
private List<Tag> tags = new ArrayList<Tag>();
@Temporal(TemporalType.DATE)
private Date date = null;
@Temporal(TemporalType.TIMESTAMP)
private Date dateCreated = new Date();
public Post() {
}
public int getIdpost() {
return this.idpost;
}
public void setIdpost(int idpost) {
this.idpost = idpost;
}
public String getContent() {
return this.content;
}
public void setContent(String content) {
this.content = content;
}
public String getTitle() {
return this.title;
}
public void setTitle(String title) {
this.title = title;
}
public User getUser() {
return this.user;
}
public void setUser(User user) {
this.user = user;
}
public List<Tag> getTags() {
return this.tags;
}
public void setTags(List<Tag> tags) {
this.tags = tags;
}
public Date getDate() {
return date;
}
public void setDate(Date date) {
this.date = date;
}
public Date getDateCreated() {
return dateCreated;
}
public void setDateCreated(Date dateCreated) {
this.dateCreated = dateCreated;
}
@Override
public String toString() {
return "Post [idpost=" + idpost + ", content=" + content + ", title="
+ title + ", date=" + date + "]";
}
}
Jest Tag podmiotu
@Entity
@NamedQueries({
@NamedQuery(name = "Tag.selectTags", query = "SELECT tag FROM Tag tag WHERE tag.tagName LIKE :keyword"),
@NamedQuery(name = "Tag.selectMatchingTags", query = "SELECT t.tagName FROM Tag t WHERE t.tagName LIKE :keyword"),
@NamedQuery(name = "Tag.selectTagByName", query = "SELECT tag FROM Tag tag WHERE tag.tagName = :tagName"),
@NamedQuery(name = "Tag.selectTagsForAllPosts", query = "SELECT DISTINCT tag FROM Tag tag, Post post JOIN tag.posts posts WHERE post.user = :user")})
public class Tag implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private int idtag;
private String tagName;
// bi-directional many-to-many association to Post
@ManyToMany(mappedBy = "tags", cascade = CascadeType.PERSIST)
private List<Post> posts;
public Tag() {
}
public Tag(String tagName) {
this.tagName = tagName;
}
public int getIdtag() {
return this.idtag;
}
public void setIdtag(int idtag) {
this.idtag = idtag;
}
public String getTagName() {
return this.tagName;
}
public void setTagName(String tagName) {
this.tagName = tagName;
}
public List<Post> getPosts() {
return this.posts;
}
public void setPosts(List<Post> posts) {
this.posts = posts;
}
@Override
public String toString() {
return tagName;
}
}
Czy możesz przesłać kod swojej jednostki organizacyjnej? Wygląda na to, że problem może się tam pojawić – kostja
@kostja Dodano kod podmiotu zgodnie z Twoimi sugestiami. i można zauważyć, że NamedQuery Tag.selectTagForAllPosts, który pobrał wszystkie znaczniki dla użytkownika, działa dobrze. Myślę, że pobieranie tagów z postów powinno być podobne, dlatego próbowałem czegoś powyżej, ale to nie działa. –
Myślę, że ta uproszczona wersja mogłaby działać: 'WYBIERZ WYBÓR p. Z p p WHERE p.tags IN: tags' – kostja