【问题标题】:Mongodb : Get Documents between specified time irrespective of datesMongodb:无论日期如何,在指定时间之间获取文档
【发布时间】:2016-02-09 22:39:18
【问题描述】:

假设MongoDB数据库中的示例文档如下:

 { "date" : ISODate("2015-11-09T05:58:19.474Z") }
 { "date" : ISODate("2014-10-25T07:30:00.241Z") }
 { "date" : ISODate("2015-11-30T15:24:00.251Z") }
 { "date" : ISODate("2012-01-10T18:36:00.101Z") }

预期:

 { "date" : ISODate("2015-11-09T05:58:19.474Z") }
 { "date" : ISODate("2014-10-25T07:30:00.241Z") }

我有兴趣查找“日期”字段中的时间在 04:15 和 07:40 之间的文档,无论日、月和年。间接查询必须匹配日期字段中的任何“YYYY-MM-DDT”。

我的方法是,从节点查询假定日期持续时间内的所有文档,然后对于与查询匹配的每个文档,将文档的“日期”字段与“yyyy-MM-DDT”+“required_time”( “YYYY-MM-DD 是从每个文档的“日期字段”中复制的,通过使用 moment.js 模块转换为 moment() 并获取月、日和年进行比较。

有什么方法可以查询直接得到同样的结果吗?

注意:我是用nodejs连接mongodb

本题基于Mongodb : Query based on time in ISODate format

【问题讨论】:

    标签: node.js mongodb mongodb-query aggregation-framework


    【解决方案1】:

    这里的基本情况是使用数学计算分钟数。注意到您已经对匹配的基本原则有了答案,但似乎在计算“分钟”部分时遇到了麻烦:

    db.collection.aggregate([
        { "$redact": {
            "$cond": {
                "if": {
                    "$and": [
                        { "$gte": [
                            { "$add": [
                                { "$hour": "$date" },
                                { "$divide": [{ "$minute": "$date" }, 60] }
                            ]},
                            4 + (15/60)
                        ]},
                        { "$lte": [
                            { "$add": [
                                { "$hour": "$date" },
                                { "$divide": [{ "$minute": "$date" }, 60] }
                            ]},
                            7 + (40/60)
                        ]}
                    ]
                },
                "then": "$$KEEP",
                "else": "$$PRUNE"
            }
        }}
    ])
    

    因此,当您将$add$minute 的除数除以$hour 时,您会得到一个仍在0-24 之间但带有小数部分的表示。

    或者直接使用日期数学,并使用$let 清理重复:

    db.collection.aggregate([
        { "$redact": {
            "$cond": {
                "if": {
                    "$let": {
                        "vars": {
                            "hours": {
                                "$divide": [
                                    { "$mod": [
                                        { "$subtract": [ "$date", new Date(0) ] },
                                        1000 * 60 * 60 * 24
                                    ]},
                                    1000 * 60 * 60
                                ]
                            }
                        },
                        "in": {
                            "$and": [
                                { "$gte": [ "$$hours", 4 + (15/60) ] },
                                { "$lte": [ "$$hours", 7 + (40/60) ] }
                            ]
                        }
                    }
                },
                "then": "$$KEEP",
                "else": "$$PRUNE"
            }
        }}
    ])
    

    无论哪种方式,这只是计算出部分组件并将其添加到选择值中。

    如果你真的想定期查询,我强烈建议你设置一个单独的文档属性,只包含“时间”值并直接查询,而不是每次都计算。

    【讨论】:

    • 聪明的想法。如果推荐任何高级的mongodb阅读材料会很棒@Blakes Seven。
    • @helloworld StackOverflow 本身怎么样?在此处阅读/搜索问题和答案,或者甚至直接在 aggregation-framework 中阅读/搜索问题和答案可能会让您对其他可能的案例和解决方案有最广泛的了解。我真的只在这里回答,但至少这给了我不同的人的问题,并提示了我白天可能没有想到的解决方案。
    • 我同意@Blakes 七。 StackOverflow 本身增加了很多知识。
    • 有人可以帮忙解决一些相关的问题吗? stackoverflow.com/questions/61131299/…
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2017-11-29
    • 1970-01-01
    • 1970-01-01
    • 2011-12-22
    • 1970-01-01
    • 2021-12-06
    • 2021-11-03
    相关资源
    最近更新 更多