建議在適當時使用屬性
更新:2007 年 11 月
型別名稱 |
UsePropertiesWhereAppropriate |
CheckId |
CA1024 |
分類 |
Microsoft.Design |
中斷變更 |
中斷 |
原因
公用或保護的方法具有以 Get 開始的名稱,該名稱不採用任何參數並且會傳回不是陣列的值。
規則描述
在大部分情況下,屬性代表資料,而方法會執行動作。存取屬性就如同存取欄位,這會讓使用屬性更為容易。如果下列其中一個條件存在,則方法是成為屬性的不錯候選者:
不採用任何引數,而且會傳回物件的狀態資訊。
接受單一引數,以設定物件狀態的某部分。
屬性應該以如同欄位般的方式運作。若方法不是這種運作方式,則不應該變更為屬性。在下列狀況中,使用方法比使用屬性更為適合:
方法用來執行較耗時的作業。方法感覺上是用來執行比設定或取得欄位值更耗時的事。
方法用來執行轉換作業。因為存取欄位並不會傳回它所儲存之資料轉換後的版本。
Get 方法有個明顯的副作用。在擷取欄位值時可避免欄位值產生額外變化。
執行的順序是很重要的。設定欄位值不會依賴其他發生的作業。
連續呼叫方法兩次會建立不同的結果。
方法是靜態 (Static),但卻會傳回呼叫端可變更的物件。擷取欄位值不允許呼叫端變更欄位所儲存的資料。
方法會傳回陣列。
如何修正違規
若要修正此規則的違規情形,請將方法變更為屬性。
隱藏警告的時機
如果方法至少會符合先前所列的其中一項準則,請隱藏這項規則的警告。
在偵錯工具中控制屬性展開
程式設計人員會避免使用屬性的一個原因是不希望偵錯工具自動展開屬性。例如,屬性可能牽涉到配置大型物件或呼叫 P/Invoke,但實際上卻不見得有任何明顯的副作用。
藉由套用 System.Diagnostics.DebuggerBrowsableAttribute,即可以防止偵錯工具自動展開屬性。下列範例顯示如何將這個屬性 (Attribute) 套用到執行個體屬性 (Property)。
Imports System
Imports System.Diagnostics
Namespace Microsoft.Samples
Public Class TestClass
' [...]
<DebuggerBrowsable(DebuggerBrowsableState.Never)> _
Public ReadOnly Property LargeObject() As LargeObject
Get
' Allocate large object
' [...]
End Get
End Property
End Class
End Namespace
using System;
using System.Diagnostics;
namespace Microsoft.Samples
{
publicclass TestClass
{
// [...]
[DebuggerBrowsable(DebuggerBrowsableState.Never)]
public LargeObject LargeObject
{
get
{
// Allocate large object
// [...]
}
}
}
範例
下列範例包含數個應轉換為屬性的方法,以及數個因行為不像欄位而不應該轉換的方法。
using System;
using System.Globalization;
using System.Collections;
namespace DesignLibrary
{
// Illustrates the behavior of rule:
// UsePropertiesWhereAppropriate.
public class Appointment
{
static long nextAppointmentID;
static double[] discountScale = {5.0, 10.0, 33.0};
string customerName;
long customerID;
DateTime when;
// Static constructor.
static Appointment()
{
// Initializes the static variable for Next appointment ID.
}
// This method will violate the rule, but should not be a property.
// This method has an observable side effect.
// Calling the method twice in succession creates different results.
public static long GetNextAvailableID()
{
nextAppointmentID++;
return nextAppointmentID - 1;
}
// This method will violate the rule, but should not be a property.
// This method performs a time-consuming operation.
// This method returns an array.
public Appointment[] GetCustomerHistory()
{
// Connect to a database to get the customer's appointment history.
return LoadHistoryFromDB(customerID);
}
// This method will violate the rule, but should not be a property.
// This method is static but returns a mutable object.
public static double[] GetDiscountScaleForUpdate()
{
return discountScale;
}
// This method will violate the rule, but should not be a property.
// This method performs a conversion.
public string GetWeekDayString()
{
return DateTimeFormatInfo.CurrentInfo.GetDayName(when.DayOfWeek);
}
// These methods will violate the rule, and should be properties.
// They each set or return a piece of the current object's state.
public DayOfWeek GetWeekDay ()
{
return when.DayOfWeek;
}
public void SetCustomerName (string customerName)
{
this.customerName = customerName;
}
public string GetCustomerName ()
{
return customerName;
}
public void SetCustomerID (long customerID)
{
this.customerID = customerID;
}
public long GetCustomerID ()
{
return customerID;
}
public void SetScheduleTime (DateTime when)
{
this.when = when;
}
public DateTime GetScheduleTime ()
{
return when;
}
// Time-consuming method that is called by GetCustomerHistory.
Appointment[] LoadHistoryFromDB(long customerID)
{
ArrayList records = new ArrayList();
// Load from database.
return (Appointment[])records.ToArray();
}
}
}