2009-10-14 18 views

Odpowiedz

48

Tak, powinieneś przetestować dwuwymiarową tablicę boolowską, aby ją głęboko skopiować. Również spojrzeć java.util.Arrays#copyOf metod, jeśli są na Java 6.

Proponuję następny kod 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

Należy pamiętać, że to nie wydają się działać dla 'Objects'. Zobacz http://stackoverflow.com/questions/15135104/system-arraycopy-copies-object-or-reference-to-object – Timo

5

Tak, to jedyny sposób, aby to zrobić. Ani java.util.Arrays nie commons-lang oferują głęboką kopię dla tablic.

6

Jestem fanem narzędzia Arrays. Ma metodę CopyOf że zrobi głęboki kopię tablicy 1-D dla ciebie, tak że chcesz coś takiego:

//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

Pamiętaj, że tworzy to tylko "głęboką kopię" dla typów pierwotnych! Sam Arrays.copyOf() tworzy tylko płytkie kopie. – codepleb

7

udało mi się wymyślić rekurencyjna tablicy głębokiej kopii. Wydaje się, że działa całkiem dobrze nawet w przypadku wielowymiarowych macierzy o różnych długościach wymiarów, np.

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

Oto metoda użytkowa.

@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: uaktualniony kod do pracy z prymitywnych tablic.

+0

To wygląda dobrze. Do czego odnosi się "Tablica"? – elgehelge

+2

Edytuj: Sam się odnalazłem: 'import java.lang.reflect.Array;' – elgehelge

2

W Javie 8 to może być realizowane jako jeden z zastosowaniem liniowej lambdas:

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

Używanie 'clone()' rzadko jest dobrym pomysłem. – Ypnypn

+2

@Ypnypn Całkowicie się zgadzam, z wyjątkiem tablic. Użycie 'clone()' jest zwykle najszybszym i najprostszym sposobem na skopiowanie tablicy. – SlavaSt

+0

'clone()' dla tablic jest całkowicie w porządku. –