The easiest way to clone an object (deep copy) in .NET is to use the serialization functions available:
1 | Public Shared Function CloneObject( ByVal obj As Object ) As Object |
2 | If Not obj Is Nothing Then |
3 | Dim bf = New System.Runtime.Serialization.Formatters.Binary.BinaryFormatter() |
4 | Dim fs = New System.IO.MemoryStream() |
6 | fs.Seek(0, System.IO.SeekOrigin.Begin) |
7 | Dim copy = CType (bf.Deserialize(fs), Object ) |
Though the performance is not very good, for occasional operations it will do the job perfectly. However, I was confronted to the following problem: what if there are events inside the class, to which other objects have subscribed? I found several methods (and functions :-)) on various places over the Internet; they basically were:
- Implement ISerializable yourself (meaning you have to update it each time you modify the class);
- Disconnect from events (retrieved using Reflection), serialize the object, and then reconnect the events (I could not make this working properly);
- Implement a serialization surrogate;
- Implement your events in a separate class that is not serialized;
- Implement your events in a C# base class.
Plenty of potential solutions, but none of them was good enough for me. So I played around with Reflection and found something that nobody else might have done so far. For a cloning interface that does just a shallow copy, like what MemberwiseClone does, but without event, I wrote this:
1 | Public Function Clone() As Object Implements System.ICloneable.Clone |
2 | Dim cl = New MyClassName( Me ) |
4 | Dim FldInfos() As Reflection.FieldInfo = Me . GetType .GetFields(Reflection.BindingFlags.Instance Or Reflection.BindingFlags. Public Or Reflection.BindingFlags.NonPublic) |
5 | For Each FldInfo As Reflection.FieldInfo In FldInfos |
6 | FldInfo.SetValue(cl, FldInfo.GetValue( Me )) |
Now if one of your class member is an object with events (or if you want to perform a deep copy), you should call its clone method (to be implemented the same way) when performing the FldInfo.SetValue, like this:
1 | For Each FldInfo As Reflection.FieldInfo In FldInfos |
2 | If FldInfo.Name <> "MyObjectWithEvents" Then |
3 | FldInfo.SetValue(cl, FldInfo.GetValue( Me )) |
5 | FldInfo.SetValue(cl, Me .MyObjectWithEvents.Clone()) |
If you have an object that is for example a dictionary of objects with events, you can call this:
1 | For Each FldInfo As Reflection.FieldInfo In FldInfos |
2 | If FldInfo.Name <> "MyObjectsWithEventsDictionary" Then |
3 | FldInfo.SetValue(cl, MyLib.CloneObject(FldInfo.GetValue( Me ))) |
5 | FldInfo.SetValue(cl, Me .MyObjectsWithEventsDictionary.ToDictionary(Of String , MyObjectWithEvent)( Function (entry) entry.Key, Function (entry) CType (entry.Value.Clone(), MyObjectWithEvent))) |
Finally, if you intend to use the Clone interface to serialize objects, you should make sure you don’t include class members marked as NonSerialized():
1 | For Each FldInfo As Reflection.FieldInfo In FldInfos |
2 | If Not FldInfo.IsNotSerialized Then |
3 | FldInfo.SetValue(cl, FldInfo.GetValue( Me )) |
I hope this will give you an insight to build something more tailored to your needs. There are other optimizations I can already think of, such as implementing a recursive Clone function where you would just put your original object and a virgin instance of it as a reference, and get a perfect serializable deep copy, whatever the class members and sub class members are! This could become a universal Clone method…