2

我有一门课,我正试图在上面运行深拷贝。此类的成员之一是“MeshContainers”,它是 MeshContainerCollection 的一个实例。

MeshContainerCollection<> 继承自我的 SceneObjectCollection<> 类,该类继承自 List<>

我注意到,源对象在 meshcontainercollection 中有 1 个项目,而克隆对象有 0。在逐步执行 DeepCopy 过程时,我注意到当我尝试获取 MeshContainerCollection 的字段时,它没有找到任何项目。现在 MeshContainerCollection 没有任何直接字段(只有继承的字段)所以我认为这是问题所在。

但我使用:

FieldInfo[] fields = type.GetFields(BindingFlags.Public |
        BindingFlags.NonPublic | BindingFlags.Instance);

其中(afaik)还应该返回私有继承成员。我已经查看了现有的 BindingFlags,但无法确定是否还有另一个 BindingFlag 我应该使用它来获取继承的私有字段。

有人能告诉我如何做一个真正的深拷贝吗?

我正在使用的深度复制方法:

private static object Process(object obj)
{
    if (obj == null)
        return null;
    Type type = obj.GetType();
    if (type.IsValueType || type == typeof(string))
    {
        return obj;
    }
    else if (type.IsArray)
    {
        Type elementType = Type.GetType(
             type.FullName.Replace("[]", string.Empty));
        var array = obj as Array;
        Array copied = Array.CreateInstance(elementType, array.Length);
        for (int i = 0; i < array.Length; i++)
        {
            copied.SetValue(Process(array.GetValue(i)), i);
        }
        return Convert.ChangeType(copied, obj.GetType());
    }
    else if (type.IsClass)
    {
        object toret = FormatterServices.GetUninitializedObject(obj.GetType());
        FieldInfo[] fields = type.GetFields(BindingFlags.Public |
                    BindingFlags.NonPublic | BindingFlags.Instance);
        foreach (FieldInfo field in fields)
        {
            object fieldValue = field.GetValue(obj);
            if (fieldValue == null)
                continue;
            field.SetValue(toret, Process(fieldValue));
        }

        return toret;
    }
    else
        throw new ArgumentException("Unknown type");
}

EDIT1:我不喜欢通过序列化而是通过反射来做到这一点。

4

2 回答 2

1

GetFields 文档中所述,

不返回基类上的私有字段。

试试这个方法:

public static IEnumerable<FieldInfo> GetAllFields(this Type type)
{
    IEnumerable<FieldInfo> fields = type.GetFields(
        BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);

    if (type.BaseType == null)
        return fields;
    else
        return GetAllFields(baseType).Concat(fields);
}

(你可能想重写它以避免所有的枚举和连接,但你明白了)

于 2012-06-19T12:36:29.060 回答
0

您可以实现此目的的方法之一是序列化对象,然后将其反序列化..编写一个函数来做同样的事情..仅供参考,您的类需要为此标记[Serializable]..

有一些库可以做同样的事情.. Copyable就是其中之一.. 我建议不要尝试重新发明轮子,而是建立在这个库上,因为当你深度复制时有太多的边缘条件需要处理..也许你也可以向作者提交补丁..

于 2012-06-19T12:30:41.433 回答