c# - ReadonlyCollection,对象是不可变的吗?

标签 c# readonly-collection immutable-collections

我正在尝试使用 ReadOnlyCollection 使对象不可变,我希望对象的属性不可变。

public ReadOnlyCollection<FooObject> MyReadOnlyList
{
    get
    {
        return new ReadOnlyCollection<FooObject>(_myDataList);
    }
}

但我有点困惑。

我尝试使用 foreach 将对象的属性更改为 MyReadOnlyList 并且...我可以更改值属性,是否正确?我了解 ReadOnlyCollection 设置了一个添加级别以使对象不可变。

最佳答案

ReadOnlyCollection 是不可变的这一事实意味着集合不能被修改,即不能在集合中添加或删除任何对象。这并不意味着它包含的对象是不可变的。

This article由 Eric Lippert 撰写,解释了不同种类的不变性是如何工作的。基本上,ReadOnlyCollection 是一个不可变外观,可以读取底层集合 (_myDataList),但不能修改它。但是,您仍然可以更改基础集合,因为您可以通过执行类似 _myDataList[0] = null 的操作来引用 _myDataList

此外,ReadOnlyCollection 返回的对象与 _myDataList 返回的对象相同,即 this._myDataList.First() == this.MyReadOnlyList.First ()(使用 LINQ)。这意味着如果 _myDataList 中的对象是可变的,那么 MyReadOnlyList 中的对象也是可变的。

如果你希望对象是不可变的,你应该相应地设计它们。例如,您可以使用:

public struct Point
{
    public Point(int x, int y)
    {
        this.X = x;
        this.Y = y;
    }

    // In C#6, the "private set;" can be removed
    public int X { get; private set; }
    public int Y { get; private set; }
}

代替:

public struct Point
{
    public int X { get; set; }
    public int Y { get; set; }
}

编辑:在这种情况下,如 Ian Goldby 所述,这两个结构都不允许您修改集合中元素的属性。发生这种情况是因为结构是值类型,当您访问元素时,集合返回值的副本。如果 Point 类型是一个类,您只能修改它的属性,这意味着返回对实际对象的引用,而不是它们值的副本。

关于c# - ReadonlyCollection,对象是不可变的吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32438988/

相关文章:

c# - 如何展平 IEnumerables 的 IEnumerables 的串联

c# - ImmutableArray 在哪里?

java - 返回 ImmutableMap 还是 Map 更好?

c# - 当不可变集合比并发更可取时

c# - 如何在 Window 10 中获取窗口内 TextBox 控件的句柄

c# - UInt32[] 与 UInt32*

c# - 用于公开成员集合的 ReadOnlyCollection 或 IEnumerable?

java - "? extends ParentClass"使只读?

c# - 如何像 CSS 附加示例中那样设置 WPF 按钮样式

c# - DataGridCheckBoxColumn 的 WPF 绑定(bind)未更新