是否可以写出类似于下面的内容?

public const string[] Titles = { "German", "Spanish", "Corrects", "Wrongs" };

当前回答

这是唯一正确的答案。目前还不能这样做。

所有其他答案都建议使用静态只读变量,类似于常量,但不相同。常量被硬编码到程序集中。静态只读变量只能设置一次,可能是在初始化对象时。

这些有时是可以互换的,但并非总是如此。

编辑:我想我应该把这个扔进来,因为似乎问这个问题的人对数组有点模糊。声明数组时,它是指向包含该数组的内存段的指针。它非常简单,因为它只是一个地址,没有复杂的逻辑控制它是否可读或可写。它给你一个指针,你可以用它做任何你想做的事情。

这就是为什么创建一个不可变数组有点棘手的部分原因。你可以写一个包装数组的类并且只允许通过返回一个副本来读取它,但这样它就不再只是一个数组了,它是一个包装数组的对象。

有些人建议使用static或readonly来模拟创建const数组时看到的行为。这些药物有一些副作用,对普通读者来说可能不太明显。同样,如果你用readonly标记一个数组,你是在标记指向数组的指针,而不是数组本身。你可以随心所欲地改变数组的内容:

public class Demo
{
    private readonly int[] _foo = new int[] {0,1,2,3};
    
    public void ChangeArray()
    {
        _foo[1] = 42; // works
        _foo = new int[] { 0, 1, 2, 3, 5 }; // won't even compile
    }
}

要真正获得const数组,需要对c#和MSIL底层代码进行更新,以允许从数组中读取,但不允许写入。本页上的其他建议是解决这个问题的各种创造性方法。

其他回答

为了完整起见,现在我们也有immutablearray供我们使用。这应该是真正不可变的:

public readonly static ImmutableArray<string> Titles =
    ImmutableArray.Create(new[] { "German", "Spanish", "Corrects", "Wrongs" });

需要System.Collections.Immutable NuGet引用。

你不能创建一个“const”数组,因为数组是对象,只能是对象 在运行时创建,const实体在编译时解析。

你可以将数组声明为“readonly”。这是 与const的效果相同,只是该值可以在运行时设置。它只能是 设置一次,之后它就是一个只读(即const)值。

.NET Framework v4.5+解决方案,改进了tdbeckett的答案:

using System.Collections.ObjectModel;

// ...

public ReadOnlyCollection<string> Titles { get; } = new ReadOnlyCollection<string>(
  new string[] { "German", "Spanish", "Corrects", "Wrongs" }
);

注意:假定集合在概念上是常量,在类级别声明它时将其设置为静态可能是有意义的。

上面的:

Initializes the property's implicit backing field once with the array. Note that { get; } - i.e., declaring only a property getter - is what makes the property itself implicitly read-only (trying to combine readonly with { get; } is actually a syntax error). Alternatively, you could just omit the { get; } and add readonly to create a field instead of a property, as in the question, but exposing public data members as properties rather than fields is a good habit to form. Creates an array-like structure (allowing indexed access) that is truly and robustly read-only (conceptually constant, once created), both with respect to: preventing modification of the collection as a whole (such as by removing or adding elements, or by assigning a new collection to the variable). preventing modification of individual elements. (Even indirect modification isn't possible - unlike with an IReadOnlyList<T> solution, where a (string[]) cast can be used to gain write access to the elements, as shown in mjepsen's helpful answer. The same vulnerability applies to the IReadOnlyCollection<T> interface, which, despite the similarity in name to class ReadOnlyCollection, does not even support indexed access, making it fundamentally unsuitable for providing array-like access.)

最好的选择:

public static readonly byte[] ZeroHash = { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 };

从c# 6开始,你可以这样写:

public static string[] Titles => new string[] { "German", "Spanish", "Corrects", "Wrongs" };

参见:c#:新的和改进的c# 6.0(特别是“表达式体函数和属性”一章)

这将使一个只读的静态属性,但它仍然允许您更改返回的数组的内容,但当您再次调用该属性时,您将再次获得原始的、未更改的数组。

为了澄清,这段代码与(或实际上是缩写)相同:

public static string[] Titles
{
    get { return new string[] { "German", "Spanish", "Corrects", "Wrongs" }; }
}

请注意,这种方法有一个缺点:每个引用实际上都会实例化一个新数组,因此如果您使用的是一个非常大的数组,这可能不是最有效的解决方案。 但是如果你重复使用相同的数组(例如把它放在一个私有属性中),它将再次打开改变数组内容的可能性。

如果你想要一个不可变数组(或列表),你也可以使用:

public static IReadOnlyList<string> Titles { get; } = new string[] { "German", "Spanish", "Corrects", "Wrongs" };

但是,这仍然有更改的风险,因为你仍然可以将其转换回字符串[]并更改内容,如下所示:

((string[]) Titles)[1] = "French";