我正在尝试获取符合以下标准的所有文件,但结果似乎不正确。
标准:发布日期与给定范围匹配且此发布时间线处于活动状态的电影
isActive=true并释放日期
每个文档都属于Movie类型,其中包含一个movieTimeline对象列表,每个movieTimeline都有一个isActive和release aseDate属性。在任何时候,单个电影时间线都可以在文档中处于活动状态。
文件的结构-
@Document
public class Movie extends BaseDocument
{
private String name;
private List<MovieTimeline> movieTimelines;
...
public class MovieTimeline
{
private Boolean isActive;
private LocalDateTime releaseDate;
private String description;
...
发布日期属性在Java中的类型为LocalDateTime,并以DB的2020-04-20T05:00:00.000 00:00格式存储。
这是我试图检索数据的代码-
@Repository
public interface MovieRepository extends MongoRepository<Movie,String>
{
List<Movie> findByMovieTimelines_ReleaseDateBetweenAndMovieTimelines_IsActive(LocalDateTime from, LocalDateTime to, boolean isActive);
}
样本数据-
{
"id": "abc",
"name": "Rango",
"movieTimelines": [{
"isActive": false,
"releaseDate": "2020-02-10T11:30:00",
"description": "ORIGINAL TIMELINE"
}, {
"isActive": true,
"releaseDate": "2020-06-15T10:30:00",
"description": "New Timeline created on 03/10/2020"
}
]
}, {
"id": "abcv",
"name": "Fight Club",
"movieTimelines": [{
"isActive": false,
"releaseDate": "2020-02-10T11:30:00",
"description": "ORIGINAL TIMELINE"
}, {
"isActive": true,
"releaseDate": "2020-05-18T10:30:00",
"description": "New Timeline created on 02/05/2020"
}
]
}, {
"id": "asd",
"name": "Death Note",
"movieTimelines": [{
"isActive": false,
"releaseDate": "2020-03-09T10:30:00",
"description": "ORIGINAL TIMELINE"
}, {
"isActive": false,
"releaseDate": "2020-03-16T10:30:00",
"description": "New Timeline created on 02/05/2020"
}, {
"isActive": false,
"releaseDate": "2020-04-20T10:30:00",
"description": "New Timeline created on 03/06/2020"
}, {
"isActive": true,
"releaseDate": "2020-04-20T10:30:00",
"description": "New Timeline created on 03/06/2020"
}
]
}, {
"id": "gfj",
"name": "Andhadhun",
"movieTimelines": [{
"isActive": false,
"releaseDate": "2020-04-13T10:30:00",
"description": "ORIGINAL TIMELINE"
}, {
"isActive": true,
"releaseDate": "2020-07-20T10:30:00",
"description": "New Timeline created on 03/09/2020"
}
]
}, {
"id": "nojh",
"name": "Evan Almighty",
"movieTimelines": [{
"isActive": true,
"releaseDate": "2020-04-20T10:30:00",
"description": "ORIGINAL TIMELINE"
}
]
}
预期-
{
"id": "asd",
"name": "Death Note",
"movieTimelines": [{
"isActive": false,
"releaseDate": "2020-03-09T10:30:00",
"description": "ORIGINAL TIMELINE"
}, {
"isActive": false,
"releaseDate": "2020-03-16T10:30:00",
"description": "New Timeline created on 02/05/2020"
}, {
"isActive": false,
"releaseDate": "2020-04-20T10:30:00",
"description": "New Timeline created on 03/06/2020"
}, {
"isActive": true,
"releaseDate": "2020-04-20T10:30:00",
"description": "New Timeline created on 03/06/2020"
}
]
},{
"id": "nojh",
"name": "Evan Almighty",
"movieTimelines": [{
"isActive": true,
"releaseDate": "2020-04-20T10:30:00",
"description": "ORIGINAL TIMELINE"
}
]
}
MongoDB是否有任何过滤器或运算符可用于在单个查询中获取这些文档?
MongoDB提供了$elemMatch投影运算符,应该能够满足您的条件。
特别是,您的文档由一系列文档组成。因此,这里是您可以用来构造此类数据查询的另一个参考。
这是我在使用您的条件和示例数据进行测试时发现的。我的查询是在Mongo Shell中,而不是Java/Spring中,所以请用所需的语言重写它。
由于您只想获取那些具有活动电影时间线的文档,因此“movieTimeline”中的“isActive”属性必须为true。使用以下查询可以仅满足此条件:
db.collectionName.find({ "movieTimelines": { $elemMatch: { "isActive": true } } })
发布日期也需要在给定的范围内,这意味着查询需要解决“movieTimeline”中的“release aseDate”属性。这个条件可以这样查询:
db.collectionName.find({ "movieTimelines": { $elemMatch: { "releaseDate": { $gte: ISODate("2020-04-20T00:00:00"), $lte: ISODate("2020-04-21T00:00:00") } } } })
或
db.collectionName.find({ "movieTimelines": { $elemMatch: { "releaseDate": { $gte: ISODate("2020-04-20"), $lte: ISODate("2020-04-21") } } } })
对于上述查询,“release aseDate”值是“ISODate()”对象,而不是我数据库中的字符串。可以使用toDate、dateFromString等聚合器将字符串转换为ISODate对象。
即使“release aseDate”值是字符串,如果条件中的数据类型更改为字符串,查询也会起作用。这可能不是一个好的做法。
db.collectionName.find({ "movieTimelines": { $elemMatch: { "releaseDate": { $gte: "2020-04-20", $lte: "2020-04-21" } } } })
因此,为了满足您的两个条件,需要将两个查询组合成一个,如下所示:
db.collectionName.find({ "movieTimelines": { $elemMatch: { "isActive": true, "releaseDate": { $gte: ISODate("2020-04-20T00:00:00"), $lte: ISODate("2020-04-21T00:00:00") } } } })
我希望这回答了你的问题。