属性概述
-
属性允许类公开获取和设置值的公共方法,而隐藏实现或验证代码。
-
get 属性访问器用于返回属性值,而 set 属性访问器用于分配新值。 这些访问器可以具有不同的访问级别。 有关详细信息,请参阅限制访问器可访问性。
-
value 关键字用于定义由
set
访问器分配的值。 -
属性可以是读-写属性(既有
get
访问器又有set
访问器)、只读属性(有get
访问器,但没有set
访问器)或只写访问器(有set
访问器,但没有get
访问器)。 只写属性很少出现,常用于限制对敏感数据的访问。 -
不需要自定义访问器代码的简单属性可以作为表达式主体定义或自动实现的属性来实现。
具有支持字段的属性
有一个实现属性的基本模式,该模式使用私有支持字段来设置和检索属性值。 get
访问器返回私有字段的值,set
访问器在向私有字段赋值之前可能会执行一些数据验证。 这两个访问器还可以在存储或返回数据之前对其执行某些转换或计算。
下面的示例阐释了此模式。 在此示例中,TimePeriod
类表示时间间隔。 在内部,该类将时间间隔以秒为单位存储在名为 _seconds
的私有字段中。 名为 Hours
的读-写属性允许客户以小时为单位指定时间间隔。 get
和 set
访问器都会执行小时与秒之间的必要转换。 此外,set
访问器还会验证数据,如果小时数无效,则引发 ArgumentOutOfRangeException。
using System;
class TimePeriod
{
private double _seconds;
public double Hours
{
get { return _seconds / 3600; }
set {
if (value < 0 || value > 24)
throw new ArgumentOutOfRangeException(
$"{nameof(value)} must be between 0 and 24.");
_seconds = value * 3600;
}
}
}
class Program
{
static void Main()
{
TimePeriod t = new TimePeriod();
// The property assignment causes the 'set' accessor to be called.
t.Hours = 24;
// Retrieving the property causes the 'get' accessor to be called.
Console.WriteLine($"Time in hours: {t.Hours}");
}
}
// The example displays the following output:
// Time in hours: 24
表达式主体定义
属性访问器通常由单行语句组成,这些语句只分配或只返回表达式的结果。 可以将这些属性作为 expression-bodied 成员来实现。 =>
符号后跟用于为属性赋值或从属性中检索值的表达式,即组成了表达式主体定义。
从 C# 6 开始,只读属性可以将 get
访问器作为 expression-bodied 成员实现。 在这种情况下,既不使用 get
访问器关键字,也不使用 return
关键字。 下面的示例将只读 Name
属性作为 expression-bodied 成员实现。
using System;
public class Person
{
private string _firstName;
private string _lastName;
public Person(string first, string last)
{
_firstName = first;
_lastName = last;
}
public string Name => $"{_firstName} {_lastName}";
}
public class Example
{
public static void Main()
{
var person = new Person("Magnus", "Hedlund");
Console.WriteLine(person.Name);
}
}
// The example displays the following output:
// Magnus Hedlund
从 C# 7.0 开始,get
和 set
访问器都可以作为 expression-bodied 成员实现。 在这种情况下,必须使用 get
和 set
关键字。 下面的示例阐释如何为这两个访问器使用表达式主体定义。 请注意,return
关键字不与 get
访问器搭配使用。
using System;
public class SaleItem
{
string _name;
decimal _cost;
public SaleItem(string name, decimal cost)
{
_name = name;
_cost = cost;
}
public string Name
{
get => _name;
set => _name = value;
}
public decimal Price
{
get => _cost;
set => _cost = value;
}
}
class Program
{
static void Main(string[] args)
{
var item = new SaleItem("Shoes", 19.95m);
Console.WriteLine($"{item.Name}: sells for {item.Price:C2}");
}
}
// The example displays output like the following:
// Shoes: sells for $19.95
自动实现的属性
在某些情况下,属性 get
和 set
访问器仅向支持字段赋值或仅从其中检索值,而不包括任何附加逻辑。 通过使用自动实现的属性,既能简化代码,还能让 C# 编译器透明地提供支持字段。
如果属性具有 get
和 set
访问器,则必须自动实现这两个访问器。 自动实现的属性通过以下方式定义:使用 get
和 set
关键字,但不提供任何实现。 下面的示例与上一个示例基本相同,只不过 Name
和 Price
是自动实现的属性。 请注意,该示例还删除了参数化构造函数,以便通过调用无参数构造函数和对象初始值设定项立即初始化 SaleItem
对象。
using System;
public class SaleItem
{
public string Name
{ get; set; }
public decimal Price
{ get; set; }
}
class Program
{
static void Main(string[] args)
{
var item = new SaleItem{ Name = "Shoes", Price = 19.95m };
Console.WriteLine($"{item.Name}: sells for {item.Price:C2}");
}
}
// The example displays output like the following:
// Shoes: sells for $19.95