Java Language                
            Klonowanie obiektów
        
        
            
    Szukaj…
Uwagi
Klonowanie może być trudne, szczególnie gdy pola obiektu zawierają inne obiekty. Są sytuacje, w których chcesz wykonać głęboką kopię zamiast kopiować tylko wartości pól (tj. Odniesienia do innych obiektów).
 Najważniejsze jest to klon jest uszkodzony i należy dwa razy pomyśleć, zanim realizacji Cloneable interfejs i przesłanianie clone metody. Metoda clone jest zadeklarowana w klasie Object a nie w interfejsie Cloneable , więc Cloneable nie działa jako interfejs, ponieważ nie ma publicznej metody clone . Rezultat jest taki, że umowa dotycząca używania clone jest słabo udokumentowana i słabo egzekwowana. Na przykład klasa zastępująca clone czasami opiera się na wszystkich klasach nadrzędnych również przesłaniających clone . Nie są do tego zmuszeni, a jeśli nie, Twój kod może zgłaszać wyjątki. 
O wiele lepszym rozwiązaniem zapewniającym funkcjonalność klonowania jest zapewnienie konstruktora kopii lub fabryki kopii . Zapoznaj się z Joshua Bloch's Effective Java Item 11: Zastąp klon ostrożnie.
Klonowanie za pomocą konstruktora kopii
Prostym sposobem klonowania obiektu jest implementacja konstruktora kopii.
public class Sheep {
    private String name;
    private int weight;
    public Sheep(String name, int weight) {
        this.name = name;
        this.weight = weight;
    }
    // copy constructor
    // copies the fields of other into the new object
    public Sheep(Sheep other) {
        this.name = other.name;
        this.weight = other.weight;
    }
}
// create a sheep
Sheep sheep = new Sheep("Dolly", 20);
// clone the sheep
Sheep dolly = new Sheep(sheep); // dolly.name is "Dolly" and dolly.weight is 20
Klonowanie poprzez wdrożenie interfejsu do klonowania
Klonowanie obiektu poprzez implementację interfejsu Cloneable .
public class Sheep implements Cloneable {
    private String name;
    private int weight;
    public Sheep(String name, int weight) {
        this.name = name;
        this.weight = weight;
    }
    @Override
    public Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
}
// create a sheep
Sheep sheep = new Sheep("Dolly", 20);
// clone the sheep
Sheep dolly =  (Sheep) sheep.clone(); // dolly.name is "Dolly" and dolly.weight is 20
Klonowanie wykonując płytką kopię
Domyślnym zachowaniem podczas klonowania obiektu jest wykonanie płytkiej kopii pól obiektu. W takim przypadku zarówno oryginalny obiekt, jak i sklonowany obiekt przechowują odwołania do tych samych obiektów.
Ten przykład pokazuje to zachowanie.
import java.util.List;
public class Sheep implements Cloneable {
    private String name;
    private int weight;
    private List<Sheep> children;
    public Sheep(String name, int weight) {
        this.name = name;
        this.weight = weight;
    }
    @Override
    public Object clone() throws CloneNotSupportedException {
        return super.clone();
    }
    public List<Sheep> getChildren() {
        return children;
    }
    public void setChildren(List<Sheep> children) {
        this.children = children;
    }
}
import java.util.Arrays;
import java.util.List;
// create a sheep
Sheep sheep = new Sheep("Dolly", 20);
// create children
Sheep child1 = new Sheep("Child1", 4);
Sheep child2 = new Sheep("Child2", 5);
sheep.setChildren(Arrays.asList(child1, child2));
// clone the sheep
Sheep dolly =  (Sheep) sheep.clone();
List<Sheep> sheepChildren = sheep.getChildren();
List<Sheep> dollysChildren = dolly.getChildren();
for (int i = 0; i < sheepChildren.size(); i++) {
    // prints true, both arrays contain the same objects
    System.out.println(sheepChildren.get(i) == dollysChildren.get(i));
}
Klonowanie wykonuje głęboką kopię
Aby skopiować zagnieżdżone obiekty, należy wykonać głęboką kopię , jak pokazano w tym przykładzie.
import java.util.ArrayList;
import java.util.List;
public class Sheep implements Cloneable {
    private String name;
    private int weight;
    private List<Sheep> children;
    public Sheep(String name, int weight) {
        this.name = name;
        this.weight = weight;
    }
    @Override
    public Object clone() throws CloneNotSupportedException {
        Sheep clone = (Sheep) super.clone();
        if (children != null) {
            // make a deep copy of the children
            List<Sheep> cloneChildren = new ArrayList<>(children.size());
            for (Sheep child : children) {
                cloneChildren.add((Sheep) child.clone());
            }
            clone.setChildren(cloneChildren);
        }
        return clone;
    }
    public List<Sheep> getChildren() {
        return children;
    }
    public void setChildren(List<Sheep> children) {
        this.children = children;
    }
}
import java.util.Arrays;
import java.util.List;
// create a sheep
Sheep sheep = new Sheep("Dolly", 20);
// create children
Sheep child1 = new Sheep("Child1", 4);
Sheep child2 = new Sheep("Child2", 5);
sheep.setChildren(Arrays.asList(child1, child2));
// clone the sheep
Sheep dolly =  (Sheep) sheep.clone();
List<Sheep> sheepChildren = sheep.getChildren();
List<Sheep> dollysChildren = dolly.getChildren();
for (int i = 0; i < sheepChildren.size(); i++) {
    // prints false, both arrays contain copies of the objects inside
    System.out.println(sheepChildren.get(i) == dollysChildren.get(i));
}
Klonowanie przy użyciu fabryki kopii
public class Sheep {
    private String name;
    
    private int weight;
    
    public Sheep(String name, int weight) {
        this.name = name;
        this.weight = weight;
    }
    
    public static Sheep newInstance(Sheep other);
        return new Sheep(other.name, other.weight)
    }
}