我正在使用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
}
}
详细说明:
在这里,我只得到&#39; name&#39;服务表的字段使用第一个查询执行(变量名=查询),然后执行第二个查询(变量名= anotherQuery),但无法获得实验室对象。
所以我得到了&#39; name&#39;和&#39; lab_LAB_ID&#39;两个字段使用第一个查询执行(变量名=查询),然后我执行第二个查询(变量名=另一个查询),所以我不能成功获得实验室对象。
我不明白这一点。 即使查询不同且变量名也不同,第二查询结果似乎依赖于第一查询结果字段。
我是对的吗?
答案 0 :(得分:0)
编辑: 问题是第一个查询的结果被缓存(在会话级别),然后在第二个查询中重用。由于您运行的初始查询未返回所有字段,因此实体仅部分初始化。当您想要重新创建实体时,请务必返回所有字段。当用户尝试重新创建仅包含属性子集的实体时,Hibernate OGM应该抛出异常。我将为此创建一个新的JIRA。
当您指定{name: 1}
时,您要求仅返回以下字段:_id
和name
;
因为您要求服务,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方言的当前行为。