3

我正在学习 C#,我遇到了以下代码

public class Album
{
     public virtual int AlbumId { get; set; }
     public virtual int GenreId { get; set; }
     public virtual int ArtistId { get; set; }
     public virtual string Title { get; set; }
     public virtual decimal Price { get; set; }
     public virtual string AlbumArtUrl { get; set; }
     public virtual Genre Genre { get; set; }
     public virtual Artist Artist { get; set; }
}

只是想知道以下有什么不同?我的意思是没有 get 和 set 你也可以访问这些公共财产。是什么让这些获取和设置变得重要?

public class Album
{
    public virtual int AlbumId;
    public virtual int GenreId;
    public virtual int ArtistId;
    public virtual string Title;
    public virtual decimal Price;
    public virtual string AlbumArtUrl;
    public virtual Genre Genre;
    public virtual Artist Artist;
}
4

6 回答 6

2

您不能声明虚拟字段

public class Album
{
  public virtual int AlbumId; // <- Syntax error
...
}

属性实际上是方法:get or(and) set,所以

public class Album
{
     public virtual int AlbumId { get; set; } // <- Both get and set methods declared as virtual ones
  ...
}

如果需要,您可以在派生类中覆盖这些 get 或(和)set:

public class GreatAlbum: Album {
    private Boolean m_IsGreat;  

    public override int AlbumId {
      get {
        if (m_IsGreat) 
          return base.AlbumId
        else
          return 0; 
      } 
      set {
        m_IsGreat = (value != 0);

        base.AlbumId = value;
      }
    }
  ... 
}
于 2013-08-06T07:04:05.310 回答
2

控制您的对象私有字段值。例如,如果您不想让整数使用空值或负值。

bool started;
    public bool Started
    {
        get { return started; }
        set
        {
            started = value;
            if (started)
                OnStarted(EventArgs.Empty);
        }

    }
another example

    int positiveNumber;

    public int PositiveNumber
    {
        get { return positiveNumber; }
        set {
            if (value < 0)
                positiveNumber = 0;
            else positiveNumber = value;
        }
    }

并且只读属性的另一种实现可能如下

   int positiveNumber;

    public int PositiveNumber
    {
        get { return positiveNumber; }

    }
于 2013-08-06T07:05:48.487 回答
1

在第一种情况下,您正在处理属性,在第二种情况下,您正在处理字段。

与使用属性相比,使用字段有几个缺点。这些缺点包括:

  1. 您可以在属性getset属性中设置断点,但不能在访问字段时设置断点。
  2. 公开字段违反了信息隐藏原则。
  3. 用于访问字段和属性的二进制 MSIL 代码是不同的,因此如果您将来将公共字段更改为公共属性,尽管源代码保持兼容,但任何依赖的二进制代码都会中断。
  4. 使用反射所需的代码是不同的,因此当您从一个字段移动到一个属性时,您的反射代码会中断。

长话短说:始终使用公共属性,切勿使用公共字段。

于 2013-08-06T07:06:37.397 回答
1

有许多不同之处:

  • 编译器将属性转换为方法。因此,您可以将它们声明为虚拟并在派生类中覆盖它们。
  • 使用属性,您可以将逻辑放入 getter 或 setter(过滤、验证等)。
  • 当您使用自动实现的属性 ( { get;set;}) 时,您似乎还不如只使用公共字段。但是,使用属性意味着您可以在以后更改您的 getter 或 setter 实现,而无需更改您的类公开的接口。如果您使用了一个字段并希望在读取该字段时实现过滤,则必须引入一种新方法,将该字段设为私有并破坏该类型的所有现有使用者。

就个人而言,我认为自动实现的属性会促进不良风格,因为它们不鼓励封装。ReSharper 等工具也喜欢使用{get;set}访问器生成属性。因此,新手开发人员通常最终会得到具有大量{get;set;}属性的类,这些属性将类型的状态暴露给世界。您至少应该{get; private set;}默认使用。

于 2013-08-06T07:12:26.567 回答
1

通过提供 get(accessor) 和 set(mutator) 方法,您可以控制访问和变异。例如:

您有一个属性,您不想将其设置为超过 15 的任何值。因此,您在 set 方法中进行了必要的限制。除非那个set方法,否则你无法控制。

但在你的例子中,你的 get 和 set 方法是默认的,意味着什么都不控制。

于 2013-08-06T07:14:29.217 回答
1

属性背后的主要原因是以受控方式保护和呈现私人数据。

实际上,属性在使用中显示了它们的能力,如下所示:

public virtual int AlbumId 
{ 
    get { // ... some magical operations ... }
    set { // ... some magical operations ... }
}

关于你的主要问题——这个例子有什么区别——要注意的重点是virtual关键字。

此关键字导致属性可覆盖,因此任何其他代码都可以覆盖默认get;方法。这意味着您有自己的默认行为,其他代码(在实体框架中非常使用)实现自己的逻辑!

你的例子中的第二个不是属性,所以它们没有表现出这种神奇的能力......!

于 2013-08-06T07:15:04.373 回答