在C#中,readonly
特性用于聲明一個(gè)只讀字段。一個(gè)只讀字段可以在聲明時(shí)或在構(gòu)造函數(shù)內(nèi)部進(jìn)行初始化,一旦被賦予了初始值,它就不能被改變。
使用
下面是使用 readonly
特性的基本語法:
public readonly int MyField;
您也可以在構(gòu)造函數(shù)中初始化只讀字段,如下所示:
public class MyClass
{
public readonly int MyField;
public MyClass(int value)
{
MyField = value;
}
}
注意事項(xiàng)
- 只讀字段只能在聲明或者構(gòu)造函數(shù)中賦值。
- 只讀字段的值不能更改,相當(dāng)于常量,但常量在編譯時(shí)需要初始化,只讀字段則在運(yùn)行時(shí)初始化。
- 只讀字段可以具有不同的值,取決于所用的構(gòu)造函數(shù)來創(chuàng)建對(duì)象。
- 靜態(tài)只讀字段在運(yùn)行時(shí)初始化,其初始化發(fā)生在靜態(tài)構(gòu)造函數(shù)中(如果存在)或者在出現(xiàn)此類的任何其他靜態(tài)成員之前。
注意:對(duì)于引用類型,readonly
修飾符僅防止修改字段本身的值,而不是防止修改字段引用的對(duì)象。換句話說,你不能更改引用字段的指向,但是可以更改該字段指向的對(duì)象的屬性或方法。
readonly修飾的字段GC如何處理它
垃圾收集器(GC)對(duì) readonly
修飾的字段無特殊處理。只讀性質(zhì)并不影響對(duì)象的垃圾回收。
垃圾回收主要基于一個(gè)對(duì)象是否還被引用來決定是否進(jìn)行回收。如果一個(gè)對(duì)象不再被任何其他對(duì)象引用,那么它就會(huì)被 GC 標(biāo)記為可回收。當(dāng) GC 運(yùn)行時(shí),這些標(biāo)記為可回收的對(duì)象將被清理掉,釋放其占用的內(nèi)存資源。
而對(duì)于 readonly
字段,它僅僅是限制了該字段的修改,也就是說一旦字段被初始化后,字段本身的值是不可以被改變的。然而這并不影響其所引用的對(duì)象在內(nèi)存中的生命周期,也不影響垃圾回收的機(jī)制。
如果一個(gè) readonly
字段所引用的對(duì)象不再被其他對(duì)象引用,那么這個(gè)對(duì)象同樣會(huì)被標(biāo)記為可回收,并在 GC 運(yùn)行時(shí)被清理。
readonly修飾的字段內(nèi)存分配在哪里
在C#中,readonly
關(guān)鍵字修飾的字段的內(nèi)存分配位置取決于它是否被聲明為靜態(tài)(static
)。
- 如果
readonly
字段是實(shí)例字段(非靜態(tài)),那么它的內(nèi)存將會(huì)在堆上分配,作為創(chuàng)建對(duì)象實(shí)例時(shí)分配的一部分。每個(gè)對(duì)象實(shí)例都有自己的readonly
實(shí)例字段副本。 - 如果
readonly
字段是靜態(tài)字段,那么它的內(nèi)存將會(huì)在高頻堆(High Frequency Heap)上分配,此處用于存儲(chǔ)所有的靜態(tài)數(shù)據(jù)。所有實(shí)例共享一個(gè)readonly
靜態(tài)字段。
無論是靜態(tài)還是非靜態(tài)的 readonly
字段,都只能在聲明時(shí)或在相應(yīng)的構(gòu)造函數(shù)中初始化。對(duì)于靜態(tài) readonly
字段,這通常發(fā)生在靜態(tài)構(gòu)造函數(shù)或者第一次引用類之前。對(duì)于非靜態(tài) readonly
字段,它們?cè)趯?shí)例構(gòu)造函數(shù)中初始化。
下面是一個(gè)代碼示例:
public class MyClass
{
public readonly int InstanceField; // 在堆上分配內(nèi)存
public static readonly int StaticField; // 在高頻堆上分配內(nèi)存
public MyClass(int value)
{
InstanceField = value;
}
// 靜態(tài)構(gòu)造函數(shù)
static MyClass()
{
StaticField = 10;
}
}
readonly線程安全
readonly
關(guān)鍵字在C#中表示一旦字段被初始化,它的值就不能再被改變。這種不可變性在某種程度上可以提高多線程環(huán)境下的線程安全性。
對(duì)于值類型(如int
、bool
、double
等)或不可變的引用類型(如string
),readonly
字段是絕對(duì)線程安全的,因?yàn)樗麄兊臓顟B(tài)一旦初始化就無法改變。
但是,對(duì)于可變的引用類型(如列表、字典或自定義類),雖然你無法改變readonly
字段本身引用的對(duì)象,但你仍然可以修改該對(duì)象的內(nèi)部狀態(tài)。例如,你可以向一個(gè)readonly
的列表中添加項(xiàng)目。如果不同的線程試圖同時(shí)修改這個(gè)列表,那么可能會(huì)遇到線程安全問題。
以下是一個(gè)例子,解釋了以上的概念:
public class MyClass
{
public readonly List<int> MyList = new List<int>(); // 可變引用類型
// ...其他代碼...
public void AddItem(int item)
{
// 需要保證線程安全,因?yàn)镸yList是可變的
lock (MyList)
{
MyList.Add(item);
}
}
}
readonly
只能保證字段本身不會(huì)被改變,而不能保證其引用的對(duì)象的狀態(tài)不被改變。在處理可變的引用類型時(shí),還需要采取額外的同步措施以確保線程安全。
readonly修飾的對(duì)象傳遞的是引用還是實(shí)例
無論字段是否被 readonly
修飾,對(duì)象的傳遞方式(引用或值)都取決于其類型。
該文章在 2023/10/30 11:16:25 編輯過