Разница между System.Array.CopyTo() и System.Array.Clone()
Какая разница между System.Array.CopyTo()
а также System.Array.Clone()
?
11 ответов
Метод 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.
Редактировать:Удалить неправильный пример.
Еще одно различие, не упомянутое до сих пор, состоит в том, что
- с
Clone()
целевой массив еще не должен существовать, поскольку новый создается с нуля. - с
CopyTo()
Мало того, что массив назначения должен уже существовать, он должен быть достаточно большим, чтобы содержать все элементы в исходном массиве из индекса, который вы указали в качестве пункта назначения.
Как указано во многих других ответах, оба метода выполняют поверхностные копии массива. Однако есть различия и рекомендации, которые еще не были учтены и выделены в следующих списках.
Характеристики 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
, Ограниченная копия гарантирует, что если операция копирования не может завершиться успешно, то состояние целевого массива не будет повреждено.
Оба исполняют мелкие копии, как сказал @PatrickDesjardins (несмотря на то, что многие заблуждающиеся души считают, что CopyTo
делает глубокую копию).
Тем не мение, CopyTo
позволяет копировать один массив в указанный индекс в целевом массиве, что дает ему значительно большую гибкость.
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"
Оба 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 ссылается на ту же область памяти, на которую ссылается соответствующий элемент в исходном массиве. Это ясно указывает на то, что для нижнего уровня не создается отдельный экземпляр. И если бы это было так, то изменение значения любого элемента в массиве Копированный или Клонированный не имело бы эффекта в соответствующем элементе исходного массива.
Я думаю, что мое объяснение является исчерпывающим, но я не нашел другого способа сделать его понятным.
Array.Clone()
будет выполнять технически глубокую копию при передаче массива int
или строка к методу в качестве ссылки.
Например
int[] numbers = new int[] { -11, 12, -42, 0, 1, 90, 68, 6, -9 };
SortByAscending(numbers); // Sort the array in ascending order by clone the numbers array to local new array.
SortByDescending(numbers); // Same as Ascending order Clone
Даже если методы сортируют массив чисел, но это не повлияет на фактическую ссылку, переданную методам сортировки. Т.е. массив чисел будет в том же несортированном исходном формате в строке № 1.
Примечание. Клонирование должно выполняться в методах сортировки.
Ответы сбивают меня с толку. Когда вы говорите мелкую копию, это означает, что они все еще указывают на один и тот же адрес. Это означает, что изменение одного из них также изменит другое.
Так что, если у меня есть A = [1,2,3,4], и я клонирую его и получаю B = [1,2,3,4]. Теперь, если я изменю B[0] = 9. Это означает, что A теперь будет A = [9,2,3,4]. Это верно?
Clone()
метод не дает ссылку на целевой экземпляр, просто дает вам копию. CopyTo()
Метод копирует элементы в существующий экземпляр.
Оба не дают ссылку на целевой экземпляр, и как многие участники говорят, что они дают поверхностную копию (иллюзию) без ссылки, это ключ.
Обе мелкие копии. Метод 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 */
Array.Clone не требует наличия целевого / целевого массива при вызове функции, тогда как Array.CopyTo требует целевой массив и индекс.
Clone()
используется для копирования только структуры данных / массива, но не копирует фактические данные.
CopyTo()
копирует структуру, а также фактические данные.
Обратите внимание: есть разница между использованием 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
}