gpt4 book ai didi

c# - 使用 Func 委托(delegate)列表对 IQueryable 执行 .Select()

转载 作者:行者123 更新时间:2023-11-30 18:10:49 25 4
gpt4 key购买 nike

首先,如果标题没有意义,请允许我道歉。我很难理解我在做什么,更不用说能够使用正确的词语来描述我在做什么了。

我正在构建一个通用网格类,在其中我通过标题/linq 表达式组合定义列:

public class Column<T>
{
public string Header { get; set; }
public Func<T, string> ValueExpression { get; set; }
}

用法:

Columns = new List<Column<Employee>>
{
new Column<Employee> {Header = "Employee Id", ValueExpression = e => e.EmployeeID.ToString()},
new Column<Employee> {Header = "Name", ValueExpression = e => e.FirstName + " " + e.LastName},
new Column<Employee> {Header = "Employee Birthday Year", ValueExpression = e => e.BirthDate.HasValue ? e.BirthDate.Value.Year.ToString() : ""}
},

我想将 ValueExpression 的 ( Func<T, string>) 转换到 IQueryable 的“员工”上:

var db = new NorthwindDataContext();
var employees = db.Employees.Select(e => e);

我可以在提取 IEnumerable<IEnumerable<string>> 时让它工作来自员工(我的观点中使用的字符串列表列表),如下所示:

var elementsList = employees.ToPagedList(PageIndex, PageSize);
var elementStringList = elementsList.ToStringList(Columns.Select(c => c.ValueExpression).ToArray());

不要介意 PagedList 的东西,它与 ToList() 无关并且有点相同;

这是 ToStringList() 扩展方法:

public static IEnumerable<IEnumerable<string>> ToStringList<T>(this IPagedList<T> enumerable, params Func<T, string>[] fields)
{
foreach (var element in enumerable)
yield return element.ToStringList(fields);
}

private static IEnumerable<string> ToStringList<T>(this T element, params Func<T, string>[] fields)
{
foreach (var field in fields)
yield return field(element);
}

问题是这种方法涉及在指定必须返回的字段之前执行 IQueryable。

因此,以下查询会在某个地方执行:

SELECT [t0].[EmployeeID], [t0].[LastName], [t0].[FirstName], [t0].[Title], [t0].[TitleOfCourtesy], [t0].[BirthDate], [t0].[HireDate], [t0].[Address], [t0].[City], [t0].[Region], [t0].[PostalCode], [t0].[Country], [t0].[HomePhone], [t0].[Extension], [t0].[Photo], [t0].[Notes], [t0].[ReportsTo], [t0].[PhotoPath]
FROM [dbo].[Employees] AS [t0]

您可能会想,从表 Employees 中检索所有字段是不希望的。

我正在寻找一种方法来以某种方式在 IQueryable 员工上构建一个扩展方法,我可以在其中传递 Func 的列表(Column 的“ValueExpression”成员)并以这种方式“构建”一个新的 IQuerable,这将执行仅从数据库中检索所需字段的 SQL。

所以我正在寻找这样的东西:

IQueryable employees = employees.SelectByExpressions(Columns.Select(c => c.ValueExpression).ToArray());

提前致谢。

最佳答案

您将遇到的基本问题是 Select 必须返回某种类型的对象,但您希望该类型具有不同的字段,具体取决于已生成的动态查询。当您使用匿名类型时,编译器会为您生成一个具有适当字段的类型。但是在您的情况下,您在编译时不知道字段列表会是什么样子,因此您无法让编译器为您生成类型。

虽然动态构建只包含所需字段的类型当然是可能的,但我不得不停下来质疑这是否有必要。使用像 LINQ-to-SQL 这样的 O/R 映射器的部分理由,特别是与数据集相反,是维护在不同时间动态返回不同字段集的代码的成本不值得您从查询中获得的少量节省或来自内存使用。事实上,在某些情况下,它甚至会使查询性能变差,因为数据库服务器无法像优化具有相同字段列表的多个查询那样优化具有不同字段列表的多个查询。

另一种选择是始终返回所有字段,但只显示选定的字段。那肯定会更简单,也更容易维护。但是,如果您已经测量了该解决方案的性能影响并确定它不满足您的要求,那么您当然可以考虑动态生成必要的类型。如果您需要有关该解决方案的帮助,我也许可以为您制作一个样本。但如果我是你,我会绝对确定在朝那个方向开始之前我需要沿着那条路走下去。

更新:这里的另一个问题是,您对这些查询的唯一目标是将它们显示在网格中,还是某种其他类型的动态绑定(bind)界面中。如果是这种情况,那么使用一种“属性包”解决方案可能会很容易,其中没有涉及特定字段的具体类型,而只是键/值对的容器,类似于 DataRow .但是,如果您尝试动态创建随后将在代码中进行操作的类型,我强烈建议您不要这样做。技术实现可能很有趣,但维护很快就会变成一场噩梦。我以前被迫维护过这样的应用程序,如果我有选择的话,我再也不会这样做了。

关于c# - 使用 Func<T, string> 委托(delegate)列表对 IQueryable 执行 .Select(),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/885197/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com