Разница между System.Array.CopyTo () и System.Array.Clone () - PullRequest
69 голосов
/ 13 октября 2008

В чем разница между System.Array.CopyTo() и System.Array.Clone()?

Ответы [ 11 ]

43 голосов
/ 13 октября 2008

Метод Clone () возвращает новый объект массива (поверхностное копирование), содержащий все элементы в исходном массиве. Метод CopyTo () копирует элементы в другой существующий массив. Оба выполняют мелкую копию. Мелкая копия означает, что содержимое (каждый элемент массива) содержит ссылки на тот же объект, что и элементы в исходном массиве. Глубокая копия (которую не выполняет ни один из этих методов) создаст новый экземпляр объекта каждого элемента, что приведет к созданию другого, но идентичного объекта.

Итак, разница:

1- CopyTo require to have a destination array when Clone return a new array.
2- CopyTo let you specify an index (if required) to the destination array.
Редактировать:

Удалить неправильный пример.

23 голосов
/ 13 октября 2008

Еще одно различие, не упомянутое до сих пор, состоит в том, что

  • с Clone() целевой массив еще не должен существовать, поскольку новый создается с нуля.
  • с CopyTo() не только должен уже существовать массив назначения, он должен быть достаточно большим, чтобы содержать все элементы в исходном массиве из индекса, который вы указали в качестве назначения.
21 голосов
/ 13 октября 2008

Оба выполняют мелкие копии, как сказал @PatrickDesjardins (несмотря на то, что многие заблуждающиеся души считают, что CopyTo делает глубокую копию).

Однако CopyTo позволяет вам копировать один массив в указанный индекс в массиве назначения, что дает ему значительно большую гибкость.

18 голосов
/ 27 августа 2012

Как указано во многих других ответах, оба метода выполняют мелкие копии массива. Однако есть различия и рекомендации, которые еще не были учтены и выделены в следующих списках.

Характеристики System.Array.Clone:

  • Тесты, использующие .NET 4.0, показывают, что он медленнее, чем CopyTo, вероятно, потому что он использует Object.MemberwiseClone;
  • Требуется привести результат к соответствующему типу;
  • Полученный массив имеет ту же длину, что и исходный.

Характеристики System.Array.CopyTo:

  • Быстрее, чем Clone при копировании в массив того же типа;
  • Он вызывает Array.Copy наследование - это возможности , наиболее полезные из них:
    • Может вставлять элементы типа значения в элементы ссылочного типа, например, копируя массив int[] в object[];
    • Может распаковывать элементы ссылочного типа в элементы типа значения, например, копируя массив object[] в штучной упаковке int в int[];
    • Может выполнять расширенные преобразования для типов значений, например, копируя int[] в long[].
    • Может понижать элементы, например, копируя массив Stream[] в MemoryStream[] (если какой-либо элемент в исходном массиве не может быть преобразован в MemoryStream, генерируется исключение).
  • Позволяет скопировать источник в целевой массив, длина которого больше, чем у источника.

Также обратите внимание, что эти методы доступны для поддержки ICloneable и ICollection, поэтому, если вы имеете дело с переменными типов массивов, вам не следует использовать Clone или CopyTo и вместо этого используйте Array.Copy или Array.ConstrainedCopy. Ограниченная копия гарантирует, что если операция копирования не может завершиться успешно, то состояние целевого массива не будет повреждено.

7 голосов
/ 06 ноября 2009
object[] myarray = new object[] { "one", 2, "three", 4, "really big number", 2324573984927361 };

//create shallow copy by CopyTo
//You have to instantiate your new array first
object[] myarray2 = new object[myarray.Length];
//but then you can specify how many members of original array you would like to copy 
myarray.CopyTo(myarray2, 0);

//create shallow copy by Clone
object[] myarray1;
//here you don't need to instantiate array, 
//but all elements of the original array will be copied
myarray1 = myarray.Clone() as object[];

//if not sure that we create a shalow copy lets test it
myarray[0] = 0;
Console.WriteLine(myarray[0]);// print 0
Console.WriteLine(myarray1[0]);//print "one"
Console.WriteLine(myarray2[0]);//print "one"

источник

2 голосов
/ 14 марта 2011

CopyTo () и Clone () делают мелкие копии. Метод Clone () создает клон исходного массива. Возвращает массив точной длины.

С другой стороны, CopyTo () копирует элементы из исходного массива в массив назначения, начиная с указанного индекса массива назначения. Обратите внимание, что это добавляет элементы к уже существующему массиву.

Следующий код будет противоречить сообщениям о том, что CopyTo () делает глубокую копию:

public class Test
{
public string s;
}

// Write Main() method and within it call test()

private void test()
{
Test[] array = new Test[1];
array[0] = new Test();
array[0].s = "ORIGINAL";

Test[] copy = new Test[1];
array.CopyTo(copy, 0);

// Next line displays "ORIGINAL"
MessageBox.Show("array[0].s = " + array[0].s);
copy[0].s = "CHANGED";

// Next line displays "CHANGED", showing that
// changing the copy also changes the original.
MessageBox.Show("array[0].s = " + array[0].s);
}

Позвольте мне объяснить это немного. Если элементы массива имеют ссылочные типы, то копия (как для Clone (), так и CopyTo ()) будет сделана до первого (верхнего) уровня. Но нижний уровень не копируется. Если нам также нужна копия более низкого уровня, мы должны сделать это явно. Вот почему после клонирования или копирования элементов ссылочного типа каждый элемент массива Cloned или Copied ссылается на ту же область памяти, на которую ссылается соответствующий элемент в исходном массиве. Это ясно указывает на то, что для нижнего уровня не создается отдельный экземпляр. И если бы это было так, то изменение значения любого элемента в массиве Копированный или Клонированный не имело бы эффекта в соответствующем элементе исходного массива.

Я думаю, что мое объяснение является исчерпывающим, но я не нашел другого способа сделать его понятным.

0 голосов
/ 13 апреля 2017

Обе мелкие копии. Метод CopyTo не является глубокой копией. Проверьте следующий код:

public class TestClass1
{
    public string a = "test1";
}

public static void ArrayCopyClone()
{
    TestClass1 tc1 = new TestClass1();
    TestClass1 tc2 = new TestClass1();

    TestClass1[] arrtest1 = { tc1, tc2 };
    TestClass1[] arrtest2 = new TestClass1[arrtest1.Length];
    TestClass1[] arrtest3 = new TestClass1[arrtest1.Length];

    arrtest1.CopyTo(arrtest2, 0);
    arrtest3 = arrtest1.Clone() as TestClass1[];

    Console.WriteLine(arrtest1[0].a);
    Console.WriteLine(arrtest2[0].a);
    Console.WriteLine(arrtest3[0].a);

    arrtest1[0].a = "new";

    Console.WriteLine(arrtest1[0].a);
    Console.WriteLine(arrtest2[0].a);
    Console.WriteLine(arrtest3[0].a);
}

/* Output is 
test1
test1
test1
new
new
new */
0 голосов
/ 09 декабря 2013

Обратите внимание: между String [] и StringBuilder [] есть разница.

В String - если вы измените String, другие скопированные нами массивы (с помощью CopyTo или Clone), которые указывают на одну и ту же строку, не изменятся, но исходный массив String будет указывать на новую строку, однако, если мы используйте StringBuilder в массиве, указатель String не изменится, поэтому он повлияет на все копии, которые мы сделали для этого массива. Например:

public void test()
{
    StringBuilder[] sArrOr = new StringBuilder[1];
    sArrOr[0] = new StringBuilder();
    sArrOr[0].Append("hello");
    StringBuilder[] sArrClone = (StringBuilder[])sArrOr.Clone();
    StringBuilder[] sArrCopyTo = new StringBuilder[1];
    sArrOr.CopyTo(sArrCopyTo,0);
    sArrOr[0].Append(" world");

    Console.WriteLine(sArrOr[0] + " " + sArrClone[0] + " " + sArrCopyTo[0]);
    //Outputs: hello world hello world hello world

    //Same result in int[] as using String[]
    int[] iArrOr = new int[2];
    iArrOr[0] = 0;
    iArrOr[1] = 1;
    int[] iArrCopyTo = new int[2];
    iArrOr.CopyTo(iArrCopyTo,0);
    int[] iArrClone = (int[])iArrOr.Clone();
    iArrOr[0]++;
    Console.WriteLine(iArrOr[0] + " " + iArrClone[0] + " " + iArrCopyTo[0]);
   // Output: 1 0 0
}
0 голосов
/ 09 июня 2013

Ответы сбивают меня с толку. Когда вы говорите мелкую копию, это означает, что они все еще указывают на один и тот же адрес. Это означает, что изменение одного из них также изменит другое.

Так что, если у меня есть A = [1,2,3,4], и я клонирую его и получаю B = [1,2,3,4]. Теперь, если я изменю B [0] = 9. Это означает, что теперь A будет A = [9,2,3,4]. Это правильно?

0 голосов
/ 01 марта 2012

Метод Clone() не дает ссылку на целевой экземпляр, просто дает вам копию. метод CopyTo() копирует элементы в существующий экземпляр.

Оба не дают ссылку на целевой экземпляр, и, как многие участники говорят, что они дают мелкую копию (иллюзорную копию) без ссылки, это ключ.

...