Auf Englisch lesen

Freigeben über


Object.MemberwiseClone Methode

Definition

Erstellt eine flache Kopie der aktuellen Object.

C#
protected object MemberwiseClone();

Gibt zurück

Eine flache Kopie der aktuellen Object.

Beispiele

Im folgenden Beispiel wird die MemberwiseClone-Methode veranschaulicht. Sie definiert eine ShallowCopy Methode, die die MemberwiseClone-Methode aufruft, um einen flachen Kopiervorgang für ein Person -Objekt auszuführen. Außerdem wird eine DeepCopy-Methode definiert, die einen Deep Copy-Vorgang für ein Person -Objekt ausführt.

C#
using System;

public class IdInfo
{
    public int IdNumber;

    public IdInfo(int IdNumber)
    {
        this.IdNumber = IdNumber;
    }
}

public class Person
{
    public int Age;
    public string Name;
    public IdInfo IdInfo;

    public Person ShallowCopy()
    {
        return (Person)MemberwiseClone();
    }

    public Person DeepCopy()
    {
        Person other = (Person)MemberwiseClone();
        other.IdInfo = new IdInfo(IdInfo.IdNumber);
        return other;
    }
}

public class Example
{
    public static void Main()
    {
        // Create an instance of Person and assign values to its fields.
        Person p1 = new()
        {
            Age = 42,
            Name = "Sam",
            IdInfo = new IdInfo(6565)
        };

        // Perform a shallow copy of p1 and assign it to p2.
        Person p2 = p1.ShallowCopy();

        // Display values of p1, p2
        Console.WriteLine("Original values of p1 and p2:");
        Console.WriteLine("   p1 instance values: ");
        DisplayValues(p1);
        Console.WriteLine("   p2 instance values:");
        DisplayValues(p2);

        // Change the value of p1 properties and display the values of p1 and p2.
        p1.Age = 32;
        p1.Name = "Frank";
        p1.IdInfo.IdNumber = 7878;
        Console.WriteLine("\nValues of p1 and p2 after changes to p1:");
        Console.WriteLine("   p1 instance values: ");
        DisplayValues(p1);
        Console.WriteLine("   p2 instance values:");
        DisplayValues(p2);

        // Make a deep copy of p1 and assign it to p3.
        Person p3 = p1.DeepCopy();
        // Change the members of the p1 class to new values to show the deep copy.
        p1.Age = 39;
        p1.Name = "George";
        p1.IdInfo.IdNumber = 8641;
        Console.WriteLine("\nValues of p1 and p3 after changes to p1:");
        Console.WriteLine("   p1 instance values: ");
        DisplayValues(p1);
        Console.WriteLine("   p3 instance values:");
        DisplayValues(p3);
    }

    public static void DisplayValues(Person p)
    {
        Console.WriteLine($"      Name: {p.Name:s}, Age: {p.Age:d}");
        Console.WriteLine($"      Value: {p.IdInfo.IdNumber:d}");
    }
}

/* The example displays the following output:
 * 
 * Original values of p1 and p2:
      p1 instance values:
         Name: Sam, Age: 42
         Value: 6565
      p2 instance values:
         Name: Sam, Age: 42
         Value: 6565

   Values of p1 and p2 after changes to p1:
      p1 instance values:
         Name: Frank, Age: 32
         Value: 7878
      p2 instance values:
         Name: Sam, Age: 42
         Value: 7878

   Values of p1 and p3 after changes to p1:
      p1 instance values:
         Name: George, Age: 39
         Value: 8641
      p3 instance values:
         Name: Frank, Age: 32
         Value: 7878
 */

In diesem Beispiel gibt die Person.IdInfo-Eigenschaft ein IdInfo-Objekt zurück. Wie die Ausgabe aus dem Beispiel zeigt, ist das geklonte MemberwiseClonePerson-Objekt eine unabhängige Kopie des ursprünglichen Objekts, mit der Ausnahme, dass sie denselben Person.IdInfo Objektverweis verwenden, wenn ein Person-Objekt geklont wird. Daher ändert das Ändern der Person.IdInfo Eigenschaft des Klons die Person.IdInfo Eigenschaft des ursprünglichen Objekts. Wenn ein Deep Copy-Vorgang ausgeführt wird, kann das geklonte Person-Objekt, einschließlich seiner Person.IdInfo-Eigenschaft, geändert werden, ohne dass sich dies auf das ursprüngliche Objekt auswirkt.

Hinweise

Mit der MemberwiseClone-Methode wird eine flache Kopie erstellt, indem ein neues Objekt erstellt und dann die nicht statischen Felder des aktuellen Objekts in das neue Objekt kopiert werden. Wenn ein Feld ein Werttyp ist, wird eine Bit-nach-Bit-Kopie des Felds ausgeführt. Wenn ein Feld ein Bezugstyp ist, wird der Verweis kopiert, aber das referenzierte Objekt ist nicht; Daher beziehen sich das ursprüngliche Objekt und dessen Klon auf dasselbe Objekt.

Um den Unterschied zwischen einem flachen und einem Deep Copy-Vorgang zu veranschaulichen, ziehen Sie ein Objekt namens X in Betracht, das auf Objekte A und B verweist. Objekt B wiederum verweist auf Objekt C. Eine flache Kopie von X erstellt ein neues Objekt X2, das auch auf Objekte A und B verweist. Im Gegensatz dazu erstellt eine tiefe Kopie von X ein neues Objekt X2, das auf die neuen Objekte A2 und B2 verweist, die Kopien von A und B. B2 wiederum auf das neue Objekt C2 verweisen, das eine Kopie von C ist.

Es gibt zahlreiche Möglichkeiten zum Implementieren eines Deep Copy-Vorgangs, wenn der flache Kopiervorgang, der von der MemberwiseClone-Methode ausgeführt wird, Ihre Anforderungen nicht erfüllt. Dazu gehören folgendes:

  • Rufen Sie einen Klassenkonstruktor des Objekts auf, das kopiert werden soll, um ein zweites Objekt mit Eigenschaftswerten zu erstellen, die aus dem ersten Objekt stammen. Dabei wird davon ausgegangen, dass die Werte eines Objekts vollständig durch den Klassenkonstruktor definiert werden.
  • Rufen Sie die MemberwiseClone-Methode auf, um eine flache Kopie eines Objekts zu erstellen, und weisen Sie dann neue Objekte zu, deren Werte dem ursprünglichen Objekt entsprechen, allen Eigenschaften oder Feldern, deren Werte Referenztypen sind. Die DeepCopy-Methode im Beispiel veranschaulicht diesen Ansatz.
  • Serialisieren Sie das Objekt so, dass es tief kopiert wird, und stellen Sie dann die serialisierten Daten in einer anderen Objektvariable wieder her.
  • Verwenden Sie Spiegelung mit Rekursion, um den Deep Copy-Vorgang auszuführen.

Gilt für:

Produkt Versionen
.NET Core 1.0, Core 1.1, Core 2.0, Core 2.1, Core 2.2, Core 3.0, Core 3.1, 5, 6, 7, 8, 9, 10
.NET Framework 1.1, 2.0, 3.0, 3.5, 4.0, 4.5, 4.5.1, 4.5.2, 4.6, 4.6.1, 4.6.2, 4.7, 4.7.1, 4.7.2, 4.8, 4.8.1
.NET Standard 1.0, 1.1, 1.2, 1.3, 1.4, 1.5, 1.6, 2.0, 2.1
UWP 10.0