C#克隆交叉引用列表

发布于 2024-12-15 01:41:45 字数 675 浏览 1 评论 0原文

我有一个“我的项目”列表。 MyItem 可能会也可能不会引用它的同级。

List<MyItem> myList = new List<MyItem>();
myList.add(...)  //add all 8 items
myList[1].RefTo = myList[3];
myList[5].RefTo = myList[2];
myList[7].RefTo = myList[5];

        Item 0
        Item 1 ----+
  +---> Item 2     |
  |     Item 3 <---+
  |     Item 4 
  +---  Item 5 <---+
        Item 6     |
        Item 7 ----+ 

我需要克隆整个列表。新列表中的每个 MyItem 都是旧列表中 MyItem 的新副本(不引用),并且新列表中的所有引用都应指向新列表中的项目。最后,即使旧列表和旧 MyItems 被完全删除,新列表也应该可以工作。

我已经在 MyItem 中实现了 ICloneable 接口,因此可以通过调用 MyItem.Clone() 来克隆该项目。但是,克隆的副本仍然引用旧列表中的 MyItems。

如何使用新列表中的对象更新 MyItems 的引用? 示例代码将受到高度赞赏。

I have a list of MyItems. MyItem may or may not refer to its peers.

List<MyItem> myList = new List<MyItem>();
myList.add(...)  //add all 8 items
myList[1].RefTo = myList[3];
myList[5].RefTo = myList[2];
myList[7].RefTo = myList[5];

        Item 0
        Item 1 ----+
  +---> Item 2     |
  |     Item 3 <---+
  |     Item 4 
  +---  Item 5 <---+
        Item 6     |
        Item 7 ----+ 

I need to make a clone of the whole list. Every MyItem in the new list is a new copy of the MyItems in the old list (not referencing) and all the references in the new list should point to the items in new list. At then end, the new list should work even the old list and the old MyItems are completely deleted.

I have implementd the ICloneable interface in MyItem so the item can be cloned by calling MyItem.Clone(). However, the cloned copy is still referencing to the MyItems in the old list.

How can I update the references of the MyItems with the objects in the new list?
Sample code would be highly appreciated.

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

慕巷 2024-12-22 01:41:45

您只需要做的就是将列表序列化到内存流,然后将其反序列化回来并创建一个克隆。由于每个对象仅在您的 RefTo 字段按照您希望的方式保留在克隆副本中(包括循环引用)后才会被序列化

namespace ConsoleApplication1
{
    [Serializable]
    class MyItem
    {
        public int MyProperty { get; set; }
        public MyItem RefTo { get; set; }
    }

    class Program
    {
        static void Main(string[] args)
        {
            List<MyItem> list1 = new List<MyItem>();
            list1.Add(new MyItem() { MyProperty = 1 });
            list1.Add(new MyItem() { MyProperty = 2 });
            list1.Add(new MyItem() { MyProperty = 3 });

            list1[1].RefTo = list1[0];
            list1[2].RefTo = list1[1];

            using (MemoryStream stream = new MemoryStream())
            {
                var bformatter = new BinaryFormatter();
                bformatter.Serialize(stream, list1);
                stream.Seek(0, SeekOrigin.Begin);
                List<MyItem> clonedCopyList = (List<MyItem>)bformatter.Deserialize(stream);
            }
        }
    }
}

What you simply do is serialize your list to a memory stream and deserialize it back and create a clone. As each object is only serialized once your RefTo field will be preserved in the cloned copy as you wish including circular references

namespace ConsoleApplication1
{
    [Serializable]
    class MyItem
    {
        public int MyProperty { get; set; }
        public MyItem RefTo { get; set; }
    }

    class Program
    {
        static void Main(string[] args)
        {
            List<MyItem> list1 = new List<MyItem>();
            list1.Add(new MyItem() { MyProperty = 1 });
            list1.Add(new MyItem() { MyProperty = 2 });
            list1.Add(new MyItem() { MyProperty = 3 });

            list1[1].RefTo = list1[0];
            list1[2].RefTo = list1[1];

            using (MemoryStream stream = new MemoryStream())
            {
                var bformatter = new BinaryFormatter();
                bformatter.Serialize(stream, list1);
                stream.Seek(0, SeekOrigin.Begin);
                List<MyItem> clonedCopyList = (List<MyItem>)bformatter.Deserialize(stream);
            }
        }
    }
}
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文