0

假设我们有以下简单的类。请注意,唯一的字段是readonly并且用于ImmutableList<int>

class Abc
{
    readonly ImmutableList<int> elts;

    public Abc(params int[] ls) => elts = ImmutableList.CreateRange(ls);
}

给定所示的构造函数,很容易从一些ints 创建一个实例:

var result_a = new Abc(10, 20, 30);

现在,我可能还想要一个可以构建Abc给定 an的构造函数IEnumerable<int>

public Abc(IEnumerable<int> ls) => elts = ImmutableList.CreateRange(ls);

所以我们的类现在看起来像这样:

class Abc
{
    readonly ImmutableList<int> elts;

    public Abc(params int[] ls) => elts = ImmutableList.CreateRange(ls);

    public Abc(IEnumerable<int> ls) => elts = ImmutableList.CreateRange(ls);
}

这确实有效:

var ls = new[] { 10, 20, 30 };

var result_b = new Abc(ls);

但是,这个构造函数有点尴尬,因为乍一看,是这样的:

new Abc(item)

可能看起来像是在Abc用单个元素 ( item) 创建一个。但是如果item实际上是一个IEnumerable<int>具有多个项目的,上面的第二个构造函数将被调用。

如果您查看 Microsoft ImmutableListAPI,它们实际上调用ImmutableList.CreateRange了类似于上面的第二个构造函数的静态方法。这很好,因为我们避免了上述视觉模糊。

好的,让我们开始为我们的Abc类绘制一个类似构造函数的简单实现:

public static Abc CreateRange(IEnumerable<int> ls)
{
    elts = ImmutableList.CreateRange(ls);

    ...
}

当然,我们在这里遇到了一个问题,因为该elts字段是readonly并且不能通过这个静态方法初始化:

在此处输入图像描述

CreateRange那么,为我们的课程实现的好方法是Abc什么?

4

2 回答 2

2

设置的唯一方法elts是调用 Abc 的构造函数。

修饰符保证设置变量的readonly唯一方法是在初始化期间readonly int i=0;或在构造函数中。

该错误是由静态方法无法访问 的实例引起的Abc,尽管您仍然无法修改它,因为它是readonly.

试试这个:

public static Abc CreateRange(IEnumerable<int> ls)
{
   return new Abc(ls);
}
于 2019-03-14T02:25:12.137 回答
1

我相信 SLaks 的建议是创建一个私有构造函数并从静态方法调用它:

private Abc(IEnumerable<int> ls) => elts = ImmutableList.CreateRange(ls);

public static Abc CreateRange(IEnumerable<int> ls) => new Abc(ls);

这样,创建实例的唯一方法(没有反射)Abc是使用静态方法,假设您没有任何公共构造函数。

于 2019-03-14T02:25:18.507 回答