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

Lyn*_*chy 10 .net c# mongodb mongodb-.net-driver

我有以下结构:

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; }
}
Run Code Online (Sandbox Code Playgroud)

我试图查询a Category以获取所有TaskName值,然后将它们返回到列表框中以显示.

我试过使用这个查询:

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

但得到的回报是:{"tasks": [{"name: "test"}]}.

无论如何只返回字符串值?

Cra*_*son 20

正如Avish所说,您必须使用聚合API来使结果文档看起来像您想要的那样.但是,如果您使用表达式树API进行项目,则驱动程序可以使其中一些消失.例如,我相信以下内容适合您:

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

这应该只tasks.name为每个类别带回一个可枚举的字符串().司机将检查此投影并仅拉回tasks.name场地.


Avi*_*ish 7

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();
Run Code Online (Sandbox Code Playgroud)

另外,您可以使用aggregation API(确实支持自定义投影,有点)来做一些花哨的事情,但这对您而言可能会过大。