spring jpa onetomany 关系@Query 不工作

spring jpa onetomany relationship @Query not working

我正在尝试通过加入提要 table 并在其 source_id 字段上进行过滤来提取所有文章记录。

我的存储库:

package com.infostream.repositories;

import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.data.jpa.repository.Query;
import org.springframework.data.repository.PagingAndSortingRepository;

import com.infostream.models.Article;
import java.lang.String;

public interface ArticleRepositoryImpl extends PagingAndSortingRepository<Article, Long> {
    Page<Article> findAll(Pageable pageRequest);

    Page<Article> findByFeedId(String feedId, Pageable pageable);

    @Query("select a from Article a join Feed f where f.source_id = ?1");
    Page<Article> findBySourceId(String sourceId, Pageable pageable);
}

Feed 模型:

package com.infostream.models;

import java.util.Date;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.Table;
import javax.validation.constraints.NotNull;

import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.infostream.serializers.JsonDateSerializer;

@Entity
@Table(name="feeds")
public class Feed extends Base {

    @Column(name = "source_id", nullable = false)
    private String sourceId;

    @Column(name = "category_id", nullable = false)
    private String categoryId;

    @NotNull
    @Column(columnDefinition="text")
    private String url;

    @Column(name = "last_visited")
    private Date lastVisited;

    public Feed() {
    }

    public Feed(String sourceId, String categoryId, String url) {
        this.sourceId = sourceId;
        this.categoryId = categoryId;
        this.url = url;
    }

    @JsonSerialize(using = JsonDateSerializer.class)
    public Date getLastVisited() {
        return lastVisited;
    }

    public void setLastVisited(Date lastVisited) {
        this.lastVisited = lastVisited;
    }

    public String getSourceId() {
        return sourceId;
    }

    public void setSourceId(String sourceId) {
        this.sourceId = sourceId;
    }

    public String getCategoryId() {
        return categoryId;
    }

    public void setCategoryId(String categoryId) {
        this.categoryId = categoryId;
    }

    public String getUrl() {
        return url;
    }

    public void setUrl(String url) {
        this.url = url;
    }



}

文章型号:

package com.infostream.models;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.Table;
import javax.validation.constraints.NotNull;

@Entity
@Table(name = "articles")
public class Article extends Base {

    public Article() {

    }

    public Article(String feedId, String title, String description, String url) {
        this.feedId = feedId;
        this.title = title;
        this.description = description;
        this.url = url;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public String getDescription() {
        return description;
    }

    public void setDescription(String description) {
        this.description = description;
    }

    public String getUrl() {
        return url;
    }

    public void setUrl(String url) {
        this.url = url;
    }

    public String getImgUrl() {
        return imgUrl;
    }

    public void setImgUrl(String imgUrl) {
        this.imgUrl = imgUrl;
    }

    public String getFeedId() {
        return feedId;
    }

    public void setFeedId(String feedId) {
        this.feedId = feedId;
    }

    @Column(name = "feed_id", nullable = false)
    private String feedId;

    @NotNull
    @Column(columnDefinition="text")
    private String title;

    @Column(name = "img_url", columnDefinition="text")
    private String imgUrl;

    @Column(columnDefinition="text")
    private String description;

    @NotNull
    @Column(columnDefinition="text")
    private String url;

    @Override
    public String toString() {
        return "Article [feedId=" + feedId + ", title=" + title + ", imgUrl=" + imgUrl + ", description=" + description
                + ", url=" + url + "]";
    }   


}

我得到的错误:

Caused by: org.hibernate.hql.internal.ast.QuerySyntaxException: Path expected for join! [select a from com.infostream.models.Article a join Feed f where f.source_id = ?1]

我曾尝试映射 OneToMany 之前得到了同样的错误,有没有人有一个很好的例子来显示它?我没有尝试过滤 feed_id。我正在过滤 source_id,这是提要 table.

上的一个字段

本质上,我想要实现的只是将这个原始 sql 查询抽象为 springs 和 hibernates 做事的方式:

select a.* from articles as a join feeds as f on(a.feed_id = f.id) where f.source_id = 'some_source_id';

经过一些修改之后,我似乎已经以一种优雅的方式解决了我的问题,并且不再需要使用 @Query 注释。我连接了所有 3 个模型(源、提要和文章)中的所有关系。 OneToMany 和 ManyToOne,然后能够使用 findBy* 生成的 spring 方法之一并且它有效。以下是所有修改后的文件,有需要的朋友可以参考。

源模型:

package com.infostream.models;

import java.util.Set;

import javax.persistence.CascadeType;
import javax.persistence.Entity;
import javax.persistence.OneToMany;
import javax.persistence.Table;
import javax.validation.constraints.NotNull;

import org.hibernate.validator.constraints.NotEmpty;

@Entity
@Table(name="sources")
public class Source extends Base {

    @NotNull
    @NotEmpty
    private String name;

    @OneToMany(mappedBy = "source", cascade = CascadeType.ALL)
    private Set<Feed> feeds;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Source() {

    }

    public Source(String name) {
        this.name = name;
    }

}

供稿模型:

package com.infostream.models;

import java.util.Date;
import java.util.Set;

import javax.persistence.CascadeType;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.OneToMany;
import javax.persistence.Table;
import javax.validation.constraints.NotNull;

import com.fasterxml.jackson.databind.annotation.JsonSerialize;
import com.infostream.serializers.JsonDateSerializer;

@Entity
@Table(name="feeds")
public class Feed extends Base {

    @ManyToOne
    @JoinColumn(name = "source_id", nullable = false)
    private Source source;


    @Column(name = "category_id", nullable = false)
    private String categoryId;

    @NotNull
    @Column(columnDefinition="text")
    private String url;

    @Column(name = "last_visited")
    private Date lastVisited;

    @OneToMany(mappedBy = "feed", cascade = CascadeType.ALL)
    private Set<Article> articles;

    public Feed() {
    }

    public Feed(Source source, String categoryId, String url) {
        this.source = source;
        this.categoryId = categoryId;
        this.url = url;
    }

    @JsonSerialize(using = JsonDateSerializer.class)
    public Date getLastVisited() {
        return lastVisited;
    }

    public void setLastVisited(Date lastVisited) {
        this.lastVisited = lastVisited;
    }

    public String getCategoryId() {
        return categoryId;
    }

    public void setCategoryId(String categoryId) {
        this.categoryId = categoryId;
    }

    public String getUrl() {
        return url;
    }

    public void setUrl(String url) {
        this.url = url;
    }



}

文章型号:

package com.infostream.models;

import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.Table;
import javax.validation.constraints.NotNull;

@Entity
@Table(name = "articles")
public class Article extends Base {

    public Article() {

    }

    public Article(Feed feed, String title, String description, String url) {
        this.feed = feed;
        this.title = title;
        this.description = description;
        this.url = url;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public String getDescription() {
        return description;
    }

    public void setDescription(String description) {
        this.description = description;
    }

    public String getUrl() {
        return url;
    }

    public void setUrl(String url) {
        this.url = url;
    }

    public String getImgUrl() {
        return imgUrl;
    }

    public void setImgUrl(String imgUrl) {
        this.imgUrl = imgUrl;
    }

    @ManyToOne
    @JoinColumn(name = "feed_id", nullable = false)
    private Feed feed;

    @NotNull
    @Column(columnDefinition="text")
    private String title;

    @Column(name = "img_url", columnDefinition="text")
    private String imgUrl;

    @Column(columnDefinition="text")
    private String description;

    @NotNull
    @Column(columnDefinition="text")
    private String url;

}

ArticleRepository 文件

package com.infostream.repositories;

import org.springframework.data.domain.Page;
import org.springframework.data.domain.Pageable;
import org.springframework.data.repository.PagingAndSortingRepository;

import com.infostream.models.Article;
import java.lang.String;

public interface ArticleRepositoryImpl extends PagingAndSortingRepository<Article, Long> {
    Page<Article> findAll(Pageable pageRequest);

    Page<Article> findByFeedId(String feedId, Pageable pageable);

    Page<Article> findByFeed_sourceId(String sourceId, Pageable pageable);
}