Подтвердить что ты не робот

Как сделать глубокую копию массива 2d в Java?

Я просто получил бит, используя .clone() в моем массиве 2d boolean, считая, что это была глубокая копия.

Как выполнить глубокую копию массива boolean[][]?

Должен ли я пропустить его и сделать серию System.arraycopy?

4b9b3361

Ответ 1

Да, вы должны перебирать 2D-булевский массив, чтобы глубоко скопировать его. Также смотрите методы java.util.Arrays#copyOf, если вы находитесь на Java 6.

Я бы предложил следующий код для Java 6:

public static boolean[][] deepCopy(boolean[][] original) {
    if (original == null) {
        return null;
    }

    final boolean[][] result = new boolean[original.length][];
    for (int i = 0; i < original.length; i++) {
        result[i] = Arrays.copyOf(original[i], original[i].length);
        // For Java versions prior to Java 6 use the next:
        // System.arraycopy(original[i], 0, result[i], 0, original[i].length);
    }
    return result;
}

Ответ 2

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

private static final int[][][] INT_3D_ARRAY = {
        {
                {1}
        },
        {
                {2, 3},
                {4, 5}
        },
        {
                {6, 7, 8},
                {9, 10, 11},
                {12, 13, 14}
        }
};

Вот метод утилиты.

@SuppressWarnings("unchecked")
public static <T> T[] deepCopyOf(T[] array) {

    if (0 >= array.length) return array;

    return (T[]) deepCopyOf(
            array, 
            Array.newInstance(array[0].getClass(), array.length), 
            0);
}

private static Object deepCopyOf(Object array, Object copiedArray, int index) {

    if (index >= Array.getLength(array)) return copiedArray;

    Object element = Array.get(array, index);

    if (element.getClass().isArray()) {

        Array.set(copiedArray, index, deepCopyOf(
                element,
                Array.newInstance(
                        element.getClass().getComponentType(),
                        Array.getLength(element)),
                0));

    } else {

        Array.set(copiedArray, index, element);
    }

    return deepCopyOf(array, copiedArray, ++index);
}

EDIT: Обновлен код для работы с примитивными массивами.

Ответ 3

Я поклонник утилиты Arrays. У него есть метод copyOf, который сделает для вас глубокую копию массива с 1-D, поэтому вам нужно что-то вроде этого:

//say you have boolean[][] foo;
boolean[][] nv = new boolean[foo.length][foo[0].length];
for (int i = 0; i < nv.length; i++)
     nv[i] = Arrays.copyOf(foo[i], foo[i].length);

Ответ 4

Да, это единственный способ сделать это. Ни java.util.Arrays not commons-lang не предлагает глубокую копию массивов.

Ответ 5

В Java 8 это может быть выполнено как однострочный с использованием lambdas:

<T> T[][] deepCopy(T[][] matrix) {
    return java.util.Arrays.stream(matrix).map(el -> el.clone()).toArray($ -> matrix.clone());
}