具有Mongodb的Hibernate OGM同一个表上的第二个查询结果依赖于同一个表的第一个查询的Result字段

时间:2018-01-17 05:37:50

标签: java spring hibernate hibernate-ogm

我正在使用Hibernate OGM(5.2.0.Alpha1)和Mongodb(3.4)

@Entity
@Table(name = "service")
@JsonInclude(Include.NON_EMPTY)
public class Service {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @Column(name = "SERVICE_ID", unique = true, nullable = false)
    @JsonSerialize(using = ToStringSerializer.class)
    public ObjectId id;

    private String name;

    @ManyToOne
    @JsonIgnore
    public Lab lab;

    getter....
    setter....
}

@Entity
@Table(name = "lab")
@JsonInclude(Include.NON_EMPTY)
// @JsonFilter(value=SalesUtils.MY_CUSTOM_FILTER_FOR_LAB)
public class Lab {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    @JsonSerialize(using = ToStringSerializer.class)
    @Column(name = "LAB_ID", unique = true, nullable = false)
    public ObjectId id;

    private String name;

    @OneToMany(mappedBy = "lab")
    public List<Service> listOfServices;

    getter....
    setter....
}

道层:

public <T> List<T> executeQuery(String query, Integer startPosition, Integer noOfRecords, T t) {
        List<T> listOfT = new ArrayList<>();

        if (SalesUtils.isObjectisNullOrEmpty(startPosition, noOfRecords)) {
            listOfT = entityManager.createNativeQuery(query.toString(), t.getClass()).getResultList();
        } else {
            listOfT = entityManager.createNativeQuery(query.toString(), t.getClass()).setFirstResult(startPosition)
                    .setMaxResults(noOfRecords).getResultList();
        }
        return SalesUtils.isListIsNullOrEmpty(listOfT) ? new ArrayList<>() : listOfT;
    }

服务层:(问题:实验室名称:null)

@Transaction
public void executeQuery(){

    String query = "db.service.find({} , {"name":1})";
    List<Service> listOfServices = myDao.executeQuery(query , null , null ,new Service());

    String anotherQuery = " { $query : { name : "CDG Service"}}";
    List<Service> listOfAnotherServices = myDao.executeQuery(query , null , null ,new Service());

    if (!SalesUtils.isListIsNullOrEmpty(listOfAnotherServices )) {
        System.out.println(listOfAnotherServices.get(0).getName());
        System.out.println(listOfAnotherServices.get(0).getLab().getName()); //null 
    }
}

服务层:(临时解决方案)

 @Transaction
    public void executeQuery(){
        //Temporary solution : added lab_LAB_ID field in below 1st query
        String query = "db.service.find({} , {"name":1,"lab_LAB_ID":1})";
        List<Service> listOfServices = myDao.executeQuery(query , null , null ,new Service());

        String anotherQuery = " { $query : { name : "CDG Service"}}";
        List<Service> listOfAnotherServices = myDao.executeQuery(query , null , null ,new Service());

        if (!SalesUtils.isListIsNullOrEmpty(listOfAnotherServices )) {
            System.out.println(listOfAnotherServices.get(0).getName());
            System.out.println(listOfAnotherServices.get(0).getLab().getName()); //not null
        }
    }

详细说明:

  1. 服务层:(问题:实验室名称:null)
  2. 在这里,我只得到&#39; name&#39;服务表的字段使用第一个查询执行(变量名=查询),然后执行第二个查询(变量名= anotherQuery),但无法获得实验室对象。

    1. 服务层:(临时解决方案)
    2. 所以我得到了&#39; name&#39;和&#39; lab_LAB_ID&#39;两个字段使用第一个查询执行(变量名=查询),然后我执行第二个查询(变量名=另一个查询),所以我不能成功获得实验室对象。

      我不明白这一点。 即使查询不同且变量名也不同,第二查询结果似乎依赖于第一查询结果字段。

      我是对的吗?

2 个答案:

答案 0 :(得分:0)

编辑: 问题是第一个查询的结果被缓存(在会话级别),然后在第二个查询中重用。由于您运行的初始查询未返回所有字段,因此实体仅部分初始化。当您想要重新创建实体时,请务必返回所有字段。当用户尝试重新创建仅包含属性子集的实体时,Hibernate OGM应该抛出异常。我将为此创建一个新的JIRA。

当您指定{name: 1}时,您要求仅返回以下字段:_idname; 因为您要求服务,Hibernate OGM会将这些值转换为Service类。问题是结果不包含实验室,因此该字段为空(应该有一个例外)。

当您运行db.service.find({} , {"name":1,"lab_LAB_ID":1})时,您还将返回实验室标识符,通过此附加信息,OGM还可以将实验室添加到服务中。

如果您需要整个服务,您应该使用的本机查询是:

    String query = "db.service.find({})";

但我建议在这种情况下使用HQL查询:

    String hqlQuery = "FROM Service";
    List<Service> services = entityManager.createQuery(hqlQuery, Service.class)
                                          .getResultList();

顺便说一下,每次运行查询时都不需要创建新的Service(),可以将参数定义为Class<T>,然后再传递Service.class

答案 1 :(得分:0)

使用本机查询,如果您选择在根类上应用投影,则必须删除toEntity,因为不应允许部分实体提取。 可能会在产品的下一个版本中引入异常抛出以防止此用途。 这也是Hibernate ORM(5.2 | 5.3)对H2方言的当前行为。