123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139 |
- /// <summary>
- /// Object extensions.
- /// This is a 3rd person class from:
- /// https://github.com/Burtsev-Alexey/net-object-deep-copy/blob/master/ObjectExtensions.cs
- /// </summary>
- using System.Collections.Generic;
- using System.Reflection;
- using System.ArrayExtensions;
- namespace System
- {
- public static class ObjectExtensions
- {
- private static readonly MethodInfo CloneMethod = typeof(Object).GetMethod("MemberwiseClone", BindingFlags.NonPublic | BindingFlags.Instance);
- public static bool IsPrimitive(this Type type)
- {
- if (type == typeof(String)) return true;
- return (type.IsValueType & type.IsPrimitive);
- }
- public static Object Copy(this Object originalObject)
- {
- return InternalCopy(originalObject, new Dictionary<Object, Object>(new ReferenceEqualityComparer()));
- }
- private static Object InternalCopy(Object originalObject, IDictionary<Object, Object> visited)
- {
- if (originalObject == null) return null;
- var typeToReflect = originalObject.GetType();
- if (IsPrimitive(typeToReflect)) return originalObject;
- if (visited.ContainsKey(originalObject)) return visited[originalObject];
- if (typeof(Delegate).IsAssignableFrom(typeToReflect)) return null;
- var cloneObject = CloneMethod.Invoke(originalObject, null);
- if (typeToReflect.IsArray)
- {
- var arrayType = typeToReflect.GetElementType();
- if (IsPrimitive(arrayType) == false)
- {
- Array clonedArray = (Array)cloneObject;
- clonedArray.ForEach((array, indices) => {
- array.SetValue(InternalCopy(clonedArray.GetValue(indices), visited), indices);
- });
- }
- }
- visited.Add(originalObject, cloneObject);
- CopyFields(originalObject, visited, cloneObject, typeToReflect);
- RecursiveCopyBaseTypePrivateFields(originalObject, visited, cloneObject, typeToReflect);
- return cloneObject;
- }
- private static void RecursiveCopyBaseTypePrivateFields(object originalObject, IDictionary<object, object> visited, object cloneObject, Type typeToReflect)
- {
- if (typeToReflect.BaseType != null)
- {
- RecursiveCopyBaseTypePrivateFields(originalObject, visited, cloneObject, typeToReflect.BaseType);
- CopyFields(originalObject, visited, cloneObject, typeToReflect.BaseType, BindingFlags.Instance | BindingFlags.NonPublic, info => info.IsPrivate);
- }
- }
- private static void CopyFields(object originalObject, IDictionary<object, object> visited, object cloneObject, Type typeToReflect, BindingFlags bindingFlags = BindingFlags.Instance | BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.FlattenHierarchy, Func<FieldInfo, bool> filter = null)
- {
- foreach (FieldInfo fieldInfo in typeToReflect.GetFields(bindingFlags))
- {
- if (filter != null && filter(fieldInfo) == false) continue;
- if (IsPrimitive(fieldInfo.FieldType)) continue;
- var originalFieldValue = fieldInfo.GetValue(originalObject);
- var clonedFieldValue = InternalCopy(originalFieldValue, visited);
- fieldInfo.SetValue(cloneObject, clonedFieldValue);
- }
- }
- public static T Copy<T>(this T original)
- {
- return (T)Copy((Object)original);
- }
- }
- public class ReferenceEqualityComparer : EqualityComparer<Object>
- {
- public override bool Equals(object x, object y)
- {
- return ReferenceEquals(x, y);
- }
- public override int GetHashCode(object obj)
- {
- if (obj == null) return 0;
- return obj.GetHashCode();
- }
- }
- namespace ArrayExtensions
- {
- public static class ArrayExtensions
- {
- public static void ForEach(this Array array, Action<Array, int[]> action)
- {
- if (array.LongLength == 0) return;
- ArrayTraverse walker = new ArrayTraverse(array);
- do action(array, walker.Position);
- while (walker.Step());
- }
- }
- internal class ArrayTraverse
- {
- public int[] Position;
- private int[] maxLengths;
- public ArrayTraverse(Array array)
- {
- maxLengths = new int[array.Rank];
- for (int i = 0; i < array.Rank; ++i)
- {
- maxLengths[i] = array.GetLength(i) - 1;
- }
- Position = new int[array.Rank];
- }
- public bool Step()
- {
- for (int i = 0; i < Position.Length; ++i)
- {
- if (Position[i] < maxLengths[i])
- {
- Position[i]++;
- for (int j = 0; j < i; j++)
- {
- Position[j] = 0;
- }
- return true;
- }
- }
- return false;
- }
- }
- }
- }
|