正如标题所示,我试图在用户登录时从用户那里获取特定数据并聚合所有数据,然后返回聚合数据。我尝试使用 $match 函数,但没有运气。令人沮丧的是我能够成功返回对象数组中的所有数据。更具体地说,下面的代码成功运行并返回一个对象数组:
const runs = await Run.find({ user: req.user.id })
但是此代码没有,它返回一个空数组:
const cumulativeTotals = await Run.aggregate([
{ $match: { user: req.user.id } },
{
$group: {
_id: null,
totalRunTime: { $sum: '$runTime' },
avgRunTime: { $avg: '$runTime' },
totalRunDistance: { $sum: '$runDistance' },
avgRunDistance: { $avg: '$runDistance' },
avgPace: { $avg: '$avgPace' },
totalHeartRate: { $avg: '$avgHeartRate' },
totalActiveCalories: { $sum: '$activeCalories' },
averageActiveCalories: { $avg: '$activeCalories' },
absoluteTotalCalories: { $sum: '$totalCalories' },
avgTotalCalories: { $avg: '$totalCalories' },
}
}
])
我不确定为什么会出现这种情况。它应该返回的是一个带有对象的数组,该对象填充了其中的所有聚合数据。
我还知道,如果删除 $match 它将成功聚合运行集合中的所有数据,所以我认为这与此有关。
以下是所有相关代码:
const getRuns = asyncHandler(async (req, res) => {
const runs = await Run.find({ user: req.user.id })
const cumulativeTotals = await Run.aggregate([
{ $match: { user: req.user.id } },
{
$group: {
_id: null,
totalRunTime: { $sum: '$runTime' },
avgRunTime: { $avg: '$runTime' },
totalRunDistance: { $sum: '$runDistance' },
avgRunDistance: { $avg: '$runDistance' },
avgPace: { $avg: '$avgPace' },
totalHeartRate: { $avg: '$avgHeartRate' },
totalActiveCalories: { $sum: '$activeCalories' },
averageActiveCalories: { $avg: '$activeCalories' },
absoluteTotalCalories: { $sum: '$totalCalories' },
avgTotalCalories: { $avg: '$totalCalories' },
}
}
])
if (!runs) {
res.status(400).json({ message: 'No Runs Found for This User' })
return
}
const response = {
runs,
cumulativeTotals
}
res.status(200).json(response)
}) Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号
因此,在 rickhg12hs 的帮助下,我们能够确定
{match: { user : req.user.id } }是我试图将字符串与 ObjectId 进行比较,这是行不通的。这是因为我设置的 API 将用户 ID 作为字符串返回,而不是作为 ObjectID。为了确保您比较的是正确的值类型(在本例中为 ObjectID),您可以这样做:{ $match: { user: new mongoose.Types.ObjectId(req.user.id) } }它获取返回的字符串并将其转换为新的 ObjectID,并且因为它现在正在比较 ObjectID,所以它可以工作。
显然,它一直在文档中,我只是没有'看起来还不够仔细