gpt4 book ai didi

c# - 公开 ReadOnlyCollection 的 ReadOnlyCollection 同时仍然能够在类中修改它的最佳方法

转载 作者:行者123 更新时间:2023-11-30 18:46:26 25 4
gpt4 key购买 nike

我不知道这是否可行,但基本上,我想公开一个 ReadOnlyCollection<ReadOnlyCollection<MyType>> 类型的属性,同时仍然能够在公开属性的类中修改基础集合。此外,我希望我的类的用户能够持有对返回集合的引用,以便它在我在公开该属性的类中内部更新时进行更新。

例如,如果这是一个单独的集合,我可以这样做:

  public class MyClass {
private List<MyType> _List;
public ReadOnlyCollection<MyType> RList {
get {
return _List.AsReadOnly();
}
}
}

这仍然允许通过执行 _List.Add() 将项目添加到我类(class)的列表中.此外,如果此类的客户要这样做,请说:

var collectionRef = myClassInstance.RList;

然后collectionRef当我从 MyClass 中向列表中添加元素时,也会发生变化.

如果我想要列表的列表,问题就会出现:

  private List<List<MyType>> _List;
public ReadOnlyCollection<ReadOnlyCollection<MyType>> RList {
get {
return _List.AsReadOnly();
}
}

上述尝试的直接问题是 AsReadonly()仅适用于外部列表。所以我会尝试返回 ReadOnlyCollection<List<MyItem>>这不是该属性的声明类型。我能想到的满足声明的属性类型的唯一方法是创建一个新的 ReadOnlyCollection。以类似于:

new ReadOnlyCollection<ReadOnlyCollection<MyItem>>(_List)

new ReadOnlyCollection<ReadOnlyCollection<MyItem>>() {
new ReadOnlyCollection<MyItem>(_List[0]),
new ReadOnlyCollection<MyItem>(_List[1]),
...
}

但是我不确定创建这个嵌套集合的语法是什么;此外,我不确定是否通过创建"new"集合,我将无法通过对属性返回值的引用来跟踪对基础列表的更改。也就是说,如果在 MyClass 里面我做_List[1].Add(myTypeInstance)我不确定持有只读版本引用的人是否会看到该新项目。

我对其他方法都持开放态度。基本上我只需要公开一个项目列表列表,它是只读的,但可以在公开属性的类中进行修改,并且客户端能够看到反射(reflect)的更改,而无需再次从属性访问器获取值。

最佳答案

ReadOnlyCollection不能做到这一点,至少不能自己做到。如果将新列表添加到 _List , RList 的任何用户需要有机会看到新的 ReadOnlyCollection链接到那个新列表。 ReadOnlyCollection根本不支持这一点。它只是原始项目的只读 View ,并且原始项目是可修改的。

您可以实现一个行为类似于 ReadOnlyCollection 的自定义类,除了它不返回原始项目,而是包装它们。 ReadOnlyCollection的实现很简单:几乎所有 ReadOnlyCollection的方法可以在一行中实现:它们要么抛出无条件异常(例如 IList.Add ),返回常量值(例如 IList.IsReadOnly ),要么转发到代理容器(例如 Count )。

为了适应您的使用,您需要进行的更改涉及直接处理项目的各种方法。请注意,这不仅仅是 this[int]索引器:您还需要确保相同列表的两个代理比较相等,并提供私有(private)/内部方法以从代理获取原始列表,以制作诸如 Contains 之类的方法。工作。您还需要创建自定义枚举器类型以确保 GetEnumerator不会开始退回原始元素。您需要创建自定义 CopyTo执行。但即使记住这些事情,也应该很容易做到。


也就是说,也许有一种不太干净但更简单的方法:如果您创建一个自定义类 MyReadOnlyCollection<T>源自 ReadOnlyCollection<T> ,您可以提供内部 Items成员(转发给 ReadOnlyCollection<T> 的 protected Items 成员)。

然后您可以创建一个 MyReadOnlyCollection<MyReadOnlyCollection<MyClass>> , 并调用 RList[0].Items.Add来自您自己的程序集,而不用担心外部用户也可以调用它。


如前所述,如果外部列表实际上永远不会改变,它可以更简单:在这种情况下,您可以简单地做一些类似的事情

public ReadOnlyCollection<ReadOnlyCollection<MyType>> RList {
get {
return _List.ConvertAll(list => list.AsReadOnly()).AsReadOnly();
}
}

哪个懒得去监控_List进行更改。

关于c# - 公开 ReadOnlyCollection 的 ReadOnlyCollection 同时仍然能够在类中修改它的最佳方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24591040/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com