使用 C# .NET 驱动程序 2.0 投影 mongodb 个子文档

Projection of mongodb subdocument using C# .NET driver 2.0

我有以下结构:

public class Category
{
    [BsonElement("name")]
    public string CategoryName { get; set; }

    [BsonDateTimeOptions]
    [BsonElement("dateCreated")]
    public DateTime DateStamp { get; set; }

    [BsonElement("tasks")]        
    public List<TaskTracker.Task> Task { get; set; }
}

public class Task
{
    [BsonElement("name")]
    public string TaskName { get; set; }

    [BsonElement("body")]
    public string TaskBody { get; set; }
}

我正在尝试查询 Category 以获取所有 TaskName 值,然后将它们 return 到要显示在列表框中的列表中。

我试过使用这个查询:

var getTasks = Categories.Find<Category>(x => x.CategoryName == catName)
                         .Project(Builders<Category>.Projection
                                                    .Include("tasks.name")
                                                    .Exclude("_id"))
                         .ToListAsync()
                         .Result;   

但是得到 returned 的是:{"tasks": [{"name: "test"}]}

是否只有 return 字符串值?

MongoDB 并不像 SQL 数据库那样真正支持预测;您可以请求部分文档,但您仍然会得到与您正在查询的文档的架构相匹配的内容。 在你的例子中,你只返回 tasks 字段,对于每个任务,只返回 name 字段。

您可以使用普通 LINQ 轻松地将其转换为字符串列表:

var categoryTasks = Categories.Find<Category>(x => x.CategoryName == catName)
                     .Project(Builders<Category>.Projection
                                                .Include("tasks.name")
                                                .Exclude("_id"))
                     .ToListAsync()
                     .Result;   

var taskNames = categoryTasks.Tasks.Select(task => task.Name).ToList();

或者,您可以使用聚合 API(确实支持自定义投影,有点)做一些奇特的事情,但这对您来说可能有点过分了。

正如 Avish 所说,您必须使用聚合 API 来使生成的文档看起来像您想要的那样。但是,如果您像对 Find 所做的那样对项目使用表达式树 API,则驱动程序可以使其中的一些消失。例如,我认为以下内容应该适合您:

var taskNames = await Categores.Find(x => x.CategoryName == catName)
    .Project(x => x.Tasks.Select(y => y.Name))
    .ToListAsync();

这应该只为每个类别返回一个可枚举的字符串 (tasks.name)。驱动程序将检查此投影并仅拉回 tasks.name 字段。