I am trying to create a deep clone of an object using the following method.
public static T DeepClone<T>(this T target)
{
using (MemoryStream stream = new MemoryStream())
{
BinaryFormatter formatter = new BinaryFormatter();
formatter.Serialize(stream, target);
stream.Position = 0;
return (T)formatter.Deserialize(stream);
}
}
This method requires an object which is Serialized i.e. an object of a class who is having an attribute "Serializable" on it. I have a class which is having attribute "DataContract" on it but the method is not working with this attribute. I think "DataContract" is also a type of serializer but maybe different than that of "Serializable".
Can anyone please give me the difference between the two? Also please let me know if it is possible to create a deepclone of an object with just 1 attribute which does the work of both "DataContract" and "Serializable" attribute or maybe a different way of creating a deepclone?
Please help!
Serializable
is needed for theBinaryFormatter
to work.DataContract
and theDataMember
attribute are used with theDataContractSerializer
.You can decorate a class with attributes for both serializers.
DataContract
is used in WCF hence .NET 3.0+. In .net 2.0 or lower there is not DataContract,DataMember
attribute, onlySerializable
.As Oded said, if you want to use BinaryFormatter you have to decorate the type with Serializable.
I once did some inspection to an object structure via Reflection to find all assemblies required for deserialization and serialize them alongside for bootstrapping.
With a bit of work one could build a similar method for deep copying. Basically you need a recursive method that carrys along a Dictionary to detect circular references. Inside the method you inspect all fields about like this:
To get it working you need to add an output object and something like
System.Runtime.Serialization.FormatterServices.GetUninitializedObject(Type type)
to create the most shallowest copy (even without copying references) of each field's value. Finally you can set each field with something likefield.SetValue(input, output)
However this implementation does not support registered event handlers, which is _un_supported by deserializing, too. Additionally each object in the hierarchy will be broken, if its class' constructor needs to initialize anything but setting all fields. The last point only work with serialization, if the class has a respective implementation, e.g. method marked
[OnDeserialized]
, implementsISerializable
,... .