4

我正在尝试延迟加载(使用收益返回的扩展)二维对象数组中的行。我收到以下错误:

c# 无法将“<>d__6”类型的对象转换为“System.Object[]”类型。

异常发生在Parse方法中找到的这一行:

yield return (TSource) conversion(o);

我不明白为什么C#认为返回值是<>d__6而不是Object[]. 我通常会编程,VB.NET所以我认为我不理解C#. 我究竟做错了什么?我查看了其他类似的问题/答案,但仍然感到困惑。

public static IEnumerable<TSource> Parse<TSource>(this object[,] array
        , Func<IEnumerable<object[]>, IEnumerable<TSource>> conversion
        , int rowStart, int columnStart, int rowCount, int columnCount)
    {

        IEnumerable<object[]> o 
            = array.ForEachRow(rowStart, columnStart, rowCount, columnCount);

        yield return (TSource) conversion(o);

    }

ForEachRow 方法:

    public static IEnumerable<object[]> ForEachRow(this object[,] array, 
               int rowStart, int columnStart, int rowCount, int columnCount)
    {

        object[] array1d=new object[columnCount];
        for (int row = rowStart; row < rowCount; row++)
        {
            for (int column = columnStart; column < columnCount; column++)
            {
                array1d[column] = array[row, column];
            }
            yield return (object[]) array1d;
        }
    }

我知道以前有人问过这个问题,但不幸的是,其他答案对我来说没有意义(我主要在 VB 中编程)。

可用于编译和测试的代码 (In VB.NET):

基本上我从 Excel 中得到一个 2D 对象数组,并想把它放在一个类中并使用 Linq 来评估。

 Dim oData As Object(,) = {{"OrderDate", "Region", "Rep", "Item", "Units", "Unit Cost", "Total"} _
        , {New DateTime(2011, 1, 6), "East", "Jones", "Pencil", 95, 1.99, 189.05} _
        , {New DateTime(2011, 1, 23), "Central", "Kivell", "Binder", 50, 19.99, 999.5} _
        , {New DateTime(2011, 2, 9), "Central", "Jardine", "Pencil", 36, 4.99, 179.64} _
        , {New DateTime(2011, 2, 26), "Central", "Gill", "Pen", 27, 19.99, 539.73} _
        , {New DateTime(2011, 3, 15), "West", "Sorvino", "Pencil", 56, 2.99, 167.44} _
        }

    Dim clsSales = oData.Parse(Of SaleOrder)(Function(o As Object()) New SaleOrder( _
                                         If(IsDate(o(0)), o(0), #1/1/1900#) _
                                         , o(1).ToString _
                                         , o(2).ToString _
                                         , o(3).ToString _
                                         , If(IsNumeric(o(4)), CInt(o(4)), 0) _
                                         , If(IsNumeric(o(5)), o(5), 0) _
                                         ), 1, 0, oData.GetUpperBound(0), 6)

    For Each cls In clsSales
        Console.WriteLine(cls.ToString)
    Next

班级在哪里:

 Class SaleOrder

Public Sub New(ByVal date_ As Date, ByVal region_ As String, ByVal rep As String, ByVal item_ As String, ByVal units As Integer _
               , ByVal cost As Double)

    OrderDate = date_
    Region = region_
    Representative = rep
    Item = item_
    UnitCount = units
    UnitCost = cost

End Sub

Public OrderDate As DateTime
Public Region As String
Public Representative As String
Public Item As String
Public UnitCount As Integer = 5
Public UnitCost As Double
Public ReadOnly Property Total() As Double
    Get
        Return UnitCount * UnitCost
    End Get
End Property

Public Overrides Function ToString() As String
    Return String.Format("{0} {1} {2} {3} {4} {5} {6}", OrderDate, Region, Representative, Item, UnitCount, UnitCost, Total)
End Function

End Class

最终解决方案

    public static IEnumerable<TSource> Parse<TSource>(this object[,] array
        , Func<object[], TSource> conversion
        , int rowStart, int columnStart, int rowCount, int columnCount)
    {

        for (int row = rowStart; row < rowCount; row++)
        {
            object[] array1d = new object[columnCount];
            for (int column = columnStart; column < columnCount; column++)
            {
                array1d[column] = array[row, column];
            }
            yield return conversion(array1d);
        }

    }
4

2 回答 2

5

有了评论中的所有信息,现在很清楚这里发生了什么。让我们做一个更简单的复制:

public static IEnumerable<Tiger> Parse()
{
  object obj = ForEachRow();
  yield return (Tiger) obj;
}

public static IEnumerable<Tiger> ForEachRow()
{
  yield return new Tiger();
}

OK,编译器对bottom方法做了什么?它像这样重写它:

class ForEachRowEnumerable : IEnumerable<Tiger>
{
    ... a class which implements an IEnumerable<Tiger> 
    that yields a single tiger...
}

public static IEnumerable<Tiger> ForEachRow()
{
  return new ForEachRowEnumerable();
}

那么现在第一种方法是做什么的呢?

你打电话给 ForEachRow。这将返回一个新的 ForEachRowEnumerable。您将其转换为对象。然后,您将对象投射到老虎身上。但是 ForEachRowEnumerable 不是老虎;这是一门会给你一系列老虎的课程。所以运行时会给出错误“无法将 ForEachRowEnumerable 转换为 Tiger”。

C# 编译器当然不会将该类命名为“ForEachRowEnumerable”。它命名它<>d__6是为了确保您实际上不可能按名称使用该类。

于 2013-01-31T21:56:45.817 回答
4

o是一个IEnumerable<object[]>

conversion(o)是一个IEnumerable<TSource>。您正在将一系列对象转换为一系列TSource项目。

然后,您将其IEnumerable<TSource>转换为TSource. 您基本上是在说,“将这一系列TSource项目视为一个TSource项目。” 运行时告诉你的是,“我不允许将项目序列视为一个TSource项目,因为它不是这样的。

您几乎可以肯定实际上想要做的只是将最后一行替换为:

return conversion(o);

您有一系列TSource项目,而这正是您需要返回的。您尝试使用迭代器块时想多了。

如果您真的非常想使用迭代器块,那么您需要产生序列中的每个项目,如下所示:

foreach (TSource item in conversion(o))
    yield return item;

但何苦呢。

于 2013-01-31T21:41:13.477 回答