繁体   English   中英

C# - 返回泛型数组类型

[英]C# - Return generic array type

如果我有一个简单的实用函数,可以将数组复制到新数组:

public static object[] CopyTo(object[] original, int startIndex, int endIndex)
{
    List<object> copied - new List<object>();
    for (int i = startIndex; i <= endIndex; i++) 
    {
        copied.Add(original[i]);
    }
    return copied.ToArray();
}

然后我希望能够像这样调用它:

int[] newThing = CopyTo(new int[] { 10, 9, 8, 7, 6 }, 2, 4);

编译器错误说cannot convert from int[] to object[] 这是意料之中的,因为我的CopyTo函数特别需要一个对象数组,而不是一个整数数组。

如何更改 CopyTo 的声明以使其动态接受并返回任何类型的数组? 我相信泛型是一种方式(尽管我对此不太熟悉)所以我尝试了:

public static T[] CopyTo(T[] original, int startIndex......)

但编译器不会将 T 识别为类型。

要使其通用,请使用以下代码:

public static T[] CopyTo<T>(T[] original, int startIndex, int endIndex)
{
    List<T> copied = new List<T>();
    for (int i = startIndex; i <= endIndex; i++) 
    {
        copied.Add(original[i]);
    }
    return copied.ToArray();
}

编辑:

顺便提一下,您也可以在不创建List<T>并将列表作为数组返回的情况下执行此操作。 只需创建一个数组(长度等于所需元素的数量)并填充它:

public static T[] CopyTo<T>(T[] original, int startIndex, int endIndex)
{
    int count = (endIndex - startIndex) + 1;
    int index = 0;
    T[] copied = new T[count];

    for (int i = startIndex; i <= endIndex; i++) 
        copied[index++] = original[i];

    return copied;
}

你也可以为它创建一个扩展方法:

public static class Extensions
{
    public static T[] CopyTo<T>(this T[] source, int start, int end)
    {
        int count = (end - start) + 1;
        int index = 0;
        T[] copied = new T[count];

        for (int i = start; i <= end; i++) 
            copied[index++] = source[i];

        return copied;
    }
}

现在你可以这样称呼它:

var original = new int[] { 10, 9, 8, 7, 6 };
var newThing = original.CopyTo(0, 2);

或者对于字符串数组:

var strOrig = "one.two.three.four.five.six.seven".Split('.');
var strNew = strOrig.CopyTo(2, 5);

尝试这个:

public static T[] CopyTo<T>(T[] original, int startIndex, int endIndex)
{
    List<T> copied = new List<T>();
    for (int i = startIndex; i < endIndex; i++) 
    {
        copied.Add(original[i]);
    }
    return copied.ToArray();
}

您不需要编写自己的函数,.NET Framework 已经内置了所有功能。您有两个选择:

  • Array.Copy(sourceArray, srcStartIndex, targetArray, tgtStartIndex, srcNumberOfElements);
  • sourceArray.CopyTo(targetArray, tgtStartIndex);

我先用一些例子来解释它们,最后我会把它组合成一个函数

public static T[] CopyFromArray<T>(this T[] sourceArray, int startIndex, int endIndex)

正如您在问题中所要求的那样(CopyTo 已被 .NET 保留,因此我将其重命名为 CopyFromArray)。

注意sourceArraytargetArray是任何类型的数组; srcStartIndex是开始指数sourceArray从,复制tgtStartIndex是开始指数targetArray复制到和srcNumberOfElements是你想从源数组复制元素的数量。

目标数组必须足够大以将所有元素复制到其中,否则会出错。 源数组的元素被直接复制到目标数组中(“原位”),这就是函数没有返回任何数据的原因(返回类型为void )。

看这个例子:

void Main()
{
    var target = new int[20];
    var srcStart=1; var tgtStart=3; var srcElements=3;
    Array.Copy((new int[] { 1,2,3,4,5,6,7,8,9 }), srcStart, 
        target, tgtStart, srcElements);
    Console.WriteLine(string.Join(",", target));
}

它工作正常并返回:

0,0,0,2,3,4,0,0,0,0,0,0,0,0,0,0,0,0,0,0

请注意, srcStart指定源数组中的索引,从该索引开始复制,而tgtStart指定目标数组中开始插入的索引,最后, srcElements指定要复制的元素数量。


请注意,还有一个更简单的版本,即.CopyTo方法。 它的工作原理如下:

void Main()
{
    var target = new int[20]; var tgtStart = 3;
    (new int[] { 1,2,3,4,5,6,7,8,9 }).CopyTo(target, tgtStart);
    Console.WriteLine(string.Join(",", target));
}

它返回:

0,0,0,1,2,3,4,5,6,7,8,9,0,0,0,0,0,0,0,0

第二个参数指定数据插入位置的索引(就像前面例子中的tgtStart一样),这就是前 3 个元素为 0 的原因。

您也可以将它与其他数据类型一起使用,例如:

    var target = new string[20];
    (new string[] { "abc", "def", "ghi" }).CopyTo(target, 3);

以同样的方式工作。


谈到您的问题,您可以使用我迄今为止提供给您的信息来组合您自己的通用扩展方法,如下所示:

void Main()
{
    int[] newThing1 = (new int[] { 10, 9, 8, 7, 6 }).CopyFromArray(2, 4);
    newThing1.Dump();

    string[] newThing2 = (new string[] { "A",  "B",  "C",  "D"}).CopyFromArray(1, 3);
    newThing2.Dump();
}

public static class Extensions
{
    public static T[] CopyFromArray<T>(this T[] sourceArray, int startIndex, int endIndex)
    {
        int numberOfElements = endIndex - startIndex + 1;
        var targetArray = new T[numberOfElements];
        Array.Copy(sourceArray, startIndex, targetArray, 0, numberOfElements);
        return targetArray;
    }
}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM