2009-10-14 7 views

Respuesta

48

Sí, debe iterar sobre una matriz booleana 2D para poder copiarla en profundidad. También mirar java.util.Arrays#copyOf métodos si se encuentra en Java 6.

que sugeriría el siguiente código para 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; 
} 
+0

Tenga en cuenta que esto no parece trabajar para 'Objects'. Consulte http://stackoverflow.com/questions/15135104/system-arraycopy-copies-object-or-reference-to-object – Timo

5

Sí, esa es la única manera de hacerlo. Ni java.util.Arrays not commons-lang ofrecen copia profunda para matrices.

6

Soy un fanático de la utilidad Arrays. Tiene un método CopyOf que va a hacer una copia profunda de una matriz 1-D para usted, por lo que querría algo así:

//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); 
+6

¡Tenga en cuenta que esto solo crea una "copia profunda" para los tipos primitivos! Arrays.copyOf() solo crea copias superficiales. – codepleb

7

He conseguido llegar a una copia profunda gama recursiva. Parece funcionar bastante bien incluso para matrices multidimensionales con diferentes longitudes de dimensión, p.

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

Aquí está el método de utilidad.

@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: Actualizado el código para trabajar con matrices primitivas.

+0

Esto se ve bien. ¿A qué se refiere 'Array'? – elgehelge

+2

Editar: Me encontré a mí mismo: 'import java.lang.reflect.Array;' – elgehelge

2

En Java 8 esto se puede lograr como una sola línea utilizando lambdas:

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

Usar 'clone()' rara vez es una buena idea. – Ypnypn

+2

@Ypnypn Estoy totalmente de acuerdo, a excepción de las matrices. El uso de 'clone()' es generalmente la manera más rápida y sencilla de copiar una matriz. – SlavaSt

+0

'clone()' para matrices está totalmente bien. –

Cuestiones relacionadas