LINQ延迟执行微妙

时间:2013-08-01 11:08:40

标签: c# winforms linq

所有,我最近本地化了整个应用程序,我遇到了以下问题;我的应用程序中有以下LINQ查询

var ccsum = from a in dtSumGL2.AsEnumerable()
    group Math.Abs(a.Field<double>(strpcPSPassBegCost)) by new 
    {
       BaseCC = a.Field<string>(strpcBaseCC)
    }
       into g
       select new
       {
          g.Key.BaseCC,
          PSPassBegCost = g.Sum()
       };

这是创建一个新对象ccsum,我们用它来创建DataTable并随后填充SQL Server数据库。

问题是每个新项目都在DataTable中创建,列名为BaseCCPSPassBegCost,但这些名称与这些名称的德语版本不匹配。现在我的问题是:有没有办法做类似的事情:

var ccsum = from a in dtSumGL2.AsEnumerable()
    group Math.Abs(a.Field<double>(strpcPSPassBegCost)) by new 
    {
       BaseCC = a.Field<string>(strpcBaseCC)
    }
       into g
       select new
       {
          g.Key.BaseCC as Resources.BaseCC,
          PSPassBegCost = g.Sum() as Resources.PSPassBegCost 
       };

以便我可以根据其本地化名称命名表格?


编辑。从DataTable检索ccsum的代码是

fooDt = Utils.LINQToDataTable(ccsum);

public static DataTable LINQToDataTable<T>(IEnumerable<T> varlist)
{

    DataTable dtReturn = new DataTable();
    PropertyInfo[] oProps = null;
    if (varlist == null) 
    return dtReturn;

    foreach (T rec in varlist)
    {
        // Use reflection to get property names, to create 
        // table, Only first time, others will follow.
        if (oProps == null)
        {
            oProps = ((Type)rec.GetType()).GetProperties();
            foreach (PropertyInfo pi in oProps)
            {
                Type colType = pi.PropertyType;
                if ((colType.IsGenericType) && (colType.GetGenericTypeDefinition() == typeof(Nullable<>)))
                {
                    colType = colType.GetGenericArguments()[0];
                }
                dtReturn.Columns.Add(new DataColumn(pi.Name, colType));
            }
        }

        DataRow dr = dtReturn.NewRow();
        foreach (PropertyInfo pi in oProps)
        {
            pi.GetValue(rec, null) == null ? DBNull.Value : pi.GetValue(rec, null);
        }
        dtReturn.Rows.Add(dr);
    }
    return dtReturn;
}

感谢您的时间。

另一种方法可能是将ccsum对象中的“列”重命名为后处理步骤,尽管在运行时请求sumcc之前不会填充{{1}}的值 - 还有其他想法吗?

3 个答案:

答案 0 :(得分:2)

创建将列索引映射到一些可读名称的类或枚举:

public static class SumGLColumn
{
    public const int BaseCC = 0;
    public const int PSPassBegCost = 1;
}

使用列索引而不是列名来查询数据表:

var ccsum = from a in dtSumGL2.AsEnumerable()
            group Math.Abs(a.Field<double>(SumGLColumn.PSPassBegCost)) 
                  by a.Field<string>(SumGLColumn.BaseCC) into g
            select new {
               BaseCCg = g.Key,
               PSPassBegCost = g.Sum()
            };

答案 1 :(得分:2)

尝试本地化系统组件以获得更好的理解是一个值得称赞的目标,但它会给您带来使用绝大多数工具和库的挑战:通常数据库工具期望这些事情保持不变并且不了解本地化。

如果您希望数据库表更容易理解,或许更实际的解决方案是生成本地化视图?视图可以存在于de模式中,并且是表的一对一翻译。这将允许您利用许多标准工具,使您的系统在内部保持一致的“中立”文化(无论您的发展文化是什么),并在任何需要的地方提供翻译。

我认为尝试将这种本地化嵌入系统​​的核心可能不值得花费大多数开发人员和工具集的期望,并且你最好提供一个外观。

答案 2 :(得分:1)

这是不可能的。在select语句中,您定义了一个匿名类型。这不是语言功能,而是编译器功能,这意味着编译器使用您定义的属性为此类型创建一个类。

这意味着编译器必须在编译时知道名称。如果你想要更动态的东西,我建议你使用字典:

select new Dictionary<string, object> 
{ 
    { Resources.BaseCC, g.Key.BaseCC },
    { Resources.PSPassBegCost , g.Sum() }
};