我有以下结构:
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
值,然后将它们返回到列表框中以显示的列表。
我尝试过使用此查询:
var getTasks = Categories.Find<Category>(x => x.CategoryName == catName)
.Project(Builders<Category>.Projection
.Include("tasks.name")
.Exclude("_id"))
.ToListAsync()
.Result;
但返回的是:{"tasks": [{"name: "test"}]}
。
无论如何只返回字符串值?
答案 0 :(得分:17)
正如Avish所说,您必须使用聚合API来使结果文档看起来像您想要的那样。但是,如果您使用表达式树API进行项目,则驱动程序可以使其中一些消失。例如,我相信以下内容适合您:
var taskNames = await Categores.Find(x => x.CategoryName == catName)
.Project(x => x.Tasks.Select(y => y.Name))
.ToListAsync();
这应该只为每个类别带回一个可枚举的字符串(tasks.name
)。驱动程序将检查此投影并仅撤回tasks.name
字段。
答案 1 :(得分:6)
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(它确实支持自定义投影,有点)做一些奇特的东西,但这可能对你来说太过分了。