CA2000:必須在超出範圍前處置物件
型別名稱 |
DisposeObjectsBeforeLosingScope |
CheckId |
CA2000 |
分類 |
Microsoft.Reliability |
中斷變更 |
中斷 |
原因
已建立 IDisposable 型別的區域物件 (Local Object),但在此物件的所有參考都超出範圍之後才會處置此物件。
規則描述
如果未在可處置物件的所有參考都超出範圍前明確處置掉該物件,則該物件將在記憶體回收行程執行該物件的完成項時,於某個不定時間被處置掉。因為可能發生例外事件,導致物件的完成項無法執行,所以應改為明確處置掉物件。
如何修正違規
若要修正此規則的違規情形,請在物件的所有參考都超出範圍之前,在此物件上呼叫 Dispose。
請注意,您可以使用 using 陳述式 (在 Visual Basic 中為 Using) 來包裝實作 IDisposable 的物件。在 using 區段結束時將會自動處置以這種方式包裝的物件。
在下列一些情況下,使用陳述式不足以保護 IDisposable 物件,而且可能會導致發生 CA2000。
要傳回可處置的物件,需要在使用中區塊外部的 try/finally 區塊中建構該物件。
初始化可處置物件的成員不應該在使用陳述式的建構函式中進行。
只被一個例外狀況處理常式保護的巢狀建構函式。例如:
using (StreamReader sr = new StreamReader(new FileStream("C:\myfile.txt", FileMode.Create))) { ... }
會使 CA2000 發生,因為 StreamReader 物件之建構中的失敗,可能會導致 FileStream 物件永遠不會關閉。
動態物件應該使用陰影物件來實作 IDisposable 物件的 Dispose 模式。
隱藏警告的時機
除非您的物件會呼叫 Dispose,而您已在該物件上呼叫方法 (例如 Close.),否則請勿隱藏這項規則的警告。
相關規則
範例
如果您要實作的方法會傳回可處置物件,請使用 try/finally 區塊而不含 Catch 區塊,來確定處置該物件。利用 try/finally 區塊,您可允許在錯誤點引發例外狀況,並確保處置該物件。
在 OpenPort1 方法中,開啟 ISerializable 物件 SerialPort 的呼叫,或是 SomeMethod 的呼叫都可能失敗。在此實作上會引發 CA2000 警告。
在 OpenPort2 方法中,會宣告兩個 SerialPort 物件並將其設定為 null:
tempPort,用來測試方法作業是否成功。
port,用於方法的傳回值。
tempPort 會在 try 區塊中建構和開啟,而且其他任何必要的工作都會在相同的 try 區塊中執行。在 try 區塊的結尾,已開啟的連接埠會指派至會傳回的 port 物件,而且 tempPort 物件會設定為 null。
finally 區塊會檢查 tempPort 的值。如果它不是 Null,方法中的作業便已失敗,而且已關閉 tempPort 以確保會釋放任何資源。如果方法的作業成功,傳回的連接埠物件將包含開啟的 SerialPort 物件,如果作業失敗,則會是 null。
Public Function OpenPort1(ByVal PortName As String) As SerialPort
Dim port As New SerialPort(PortName)
port.Open() 'CA2000 fires because this might throw
SomeMethod() 'Other method operations can fail
Return port
End Function
Public Function OpenPort2(ByVal PortName As String) As SerialPort
Dim tempPort As SerialPort = Nothing
Dim port As SerialPort = Nothing
Try
tempPort = New SerialPort(PortName)
tempPort.Open()
SomeMethod()
'Add any other methods above this line
port = tempPort
tempPort = Nothing
Finally
If Not tempPort Is Nothing Then
tempPort.Close()
End If
End Try
Return port
End Function
Public Function CreateReader1(ByVal x As Integer) As StreamReader
Dim local As New StreamReader("C:\Temp.txt")
x += 1
Return local
End Function
Public Function CreateReader2(ByVal x As Integer) As StreamReader
Dim local As StreamReader = Nothing
Dim localTemp As StreamReader = Nothing
Try
localTemp = New StreamReader("C:\Temp.txt")
x += 1
local = localTemp
localTemp = Nothing
Finally
If (Not (localTemp Is Nothing)) Then
localTemp.Dispose()
End If
End Try
Return local
End Function
public SerialPort OpenPort1(string portName)
{
SerialPort port = new SerialPort(portName);
port.Open(); //CA2000 fires because this might throw
SomeMethod(); //Other method operations can fail
return port;
}
public SerialPort OpenPort2(string portName)
{
SerialPort tempPort = null;
SerialPort port = null;
try
{
tempPort = new SerialPort(portName);
tempPort.Open();
SomeMethod();
//Add any other methods above this line
port = tempPort;
tempPort = null;
}
finally
{
if (tempPort != null)
{
tempPort.Close();
}
}
return port;
}
預設情況下,Visual Basic 編譯器會對所有算術運算子進行溢位檢查。因此,任何 Visual Basic 算術運算都可能擲回 OverflowException。這可能會導致如 CA2000 規則中的意外違規。例如,下列 CreateReader1 函式會產生 CA2000 違規,因為 Visual Basic 編譯器會針對可能擲回例外狀況,進而使 StreamReader 不處置的加入,發出溢位檢查指令。
若要修正此問題,您可以用您專案中的 Visual Basic 編譯器來停用的溢位檢查的發出,或者可以像下列 CreateReader2 函式一樣修改您的程式碼。
若要停用發出溢位檢查,請以滑鼠右鍵按一下 [方案總管] 中的專案名稱,然後按一下 [內容]。請按一下 [編譯],按一下 [進階編譯選項],然後再核取 [移除整數的溢位檢查]。