1

我有两个集合:数据和用户。在数据集合中,有一个由大约 300 到 800 个用户组成的用户 ID 数组。

我需要将数据集合中每一行的所有用户的国家/地区连接在一起,由于一次查询的数据太多,这会挂起我的网络浏览器。

我一次查询了大约 16 行 Data 集合,到目前为止,Users 集合中有 18833 个用户。

到目前为止,我已经尝试为 Meteor 集合创建 Meteor 方法和 transform() JOIN,这就是挂起我的应用程序的原因。

蒙戈系列:

UserInfo = new Mongo.Collection("userInfo")
GlyphInfo = new Mongo.Collection("GlyphAllinOne", {
    transform: function(doc) {
        doc.peopleInfo = doc.peopleInfo.forEach(function(person) {  
            person.code3 = UserInfo.findOne({userId: person.name}).code3;
            return person;
        })
        return doc;
    }
});

'code3' 指定用户的国家。

出版物:

Meteor.publish("glyphInfo", function (courseId) {
    this.unblock();
    var query = {};
    if (courseId) query.courseId = courseId;
    return [GlyphInfo.find(query), UserInfo.find({})];
})

测试服务器方法:

Meteor.methods({
    'glyph.countryDistribution': function(courseId) {
        var query = {};
        if (courseId) query.courseId = courseId;
        var glyphs = _.map(_.pluck(GlyphInfo.find(query).fetch(), 'peopleInfo'), function(glyph) {
            _.map(glyph, function(user) {
                var data = Users.findOne({userId: user.name});
                if (data) {
                    user.country = data ? data.code3 : null;
                    console.log(user.country)
                    return user;
                }
            });
            return glyph;
        });
        return glyphs;
    }
});

采集数据:

GlyphAllInOne 集合 用户信息集合

有一个预处理我的收藏的选项,以便已经包含国家/地区,但是我不允许修改这些收藏。我认为在服务器启动时完成此 JOIN 并随后通过数组作为 Meteor 方法公开可能会使服务器的启动时间延迟太久;虽然我不确定。

有人对如何加快此查询有任何想法吗?

编辑:也尝试了 MongoDB 聚合命令,它在 Meteor 的 minimongo 上似乎非常慢。与本机 MongoDB 客户端上的 1 秒相比,查询需要 4 分钟。

var codes = GlyphInfo.aggregate([

        {$unwind: "$peopleInfo"},
        {$lookup: {
            from: "users",
            localField: "peopleInfo.name",
            foreignField: "userId",
            as: "details"
        }
        },
        {$unwind: "$details"},
        {$project: {"peopleInfo.Count": 1, "details.code3": 1}}
    ])
4

2 回答 2

0

我会稍微不同地处理这个问题,使用reywood:publish-composite

  1. 在服务器上,我将使用 publish-composite 发布 glyphinfo,然后在发布中包含相关用户及其国家/地区字段
  2. 在需要显示国家名称和 glyphinfo 对象的任何地方,我都会在客户端加入国家/地区

出版物:

Meteor.publishComposite('glyphInfo', function(courseId) {
    this.unblock();
    return {
        find: function() {
        var query = {};
        if (courseId) query.courseId = courseId;
        return GlyphInfo.find(query);
        },
        children: [
            {
                find: function(glyph) {
                    var nameArray = [];
                    glyph.person.forEach(function(person){
                      nameArray.push(person.name);
                    };
                    return UserInfo.find({ userId: {$in: nameArray }});
            }
        ]
    }
});
于 2016-07-12T17:56:48.410 回答
0

通过创建一个巨大的 MongoDB 聚合调用解决了这个问题,解决延迟的最大因素是索引数据库中的唯一列。

在我的数据库中仔细实现了超过 460 万个条目的索引后,在 Robomongo 上花了 0.3 秒,在 Meteor 上向客户端发送数据需要 1.4 秒。

以下是那些想看的人的聚合代码:

Meteor.methods({
    'course.countryDistribution': function (courseId, videoId) {
        var query = {};
        if (courseId) query.courseId = courseId;

        var data = GlyphInfo.aggregate([

            {$unwind: "$peopleInfo"},
            {$lookup: {
                from: "users",
                localField: "peopleInfo.name",
                foreignField: "userId",
                as: "details"
            }
            },
            {$unwind: "$details"},
            {$project: {"peopleInfo.Count": 1, "details.code3": 1}},
            {$group: {_id: "$details.code3", count: {$sum: "$peopleInfo.Count"}}}
        ])

        return data;
    }
});

如果其他人正在处理类似的问题,请随时与我联系。感谢大家的支持!

于 2016-07-13T07:01:09.427 回答