Effective c#学习笔记(1)

条款1 使用属性代替可访问的数据成员

 属性是个啥,谁在用?  

    1. C#的属性在被访问的时候看起来好像是数据成员,但其实是方法。
    2. 在.NET框架中,一般使用属性表达公有数据成员。
    3. .NET框架中的数据绑定类支持属性。

属性比公有数据成员有啥好?

    1. 随着时间的推移,新的需求影响原来类型的实现。比如在员工管理系统中有如下代码:

Effective c#学习笔记(1)
   public class Person
        {
            private string name;
            public string Name
            {
                get{return name;}
                set{this.name = value;}
            }
            ....//其它属性
        }     
View Code

       Person类声明了一个Name的属性,表示这个人的名字。程序运行了一段时间后,需求发生了变化,要求Name必须不为null或者empty。

      我们一般的做法会是1)在界面输入的地方增加验证;2)写一个单独的程序或者sql修改数据库的数据。但问题来了如果这个Person类有好几个录入的界面,比如员工的增加界面、员工修改界面,在修改的时候就要修改两次。大家肯定发现了,如果有很多这样界面,那么就需要修改很多这样的地方。再比如Person类被很多子类继承了,比如Leader、BussinessMan等,那么就需要在每个用到这些类地方进行修改。幸亏我们写的是C#,幸亏你用的是属性。只需要完成以下修改就可以到达我们的目的了。

Effective c#学习笔记(1)
   public class Person
       {
            private string name;
            public string Name
            {
                get{return name;}
                set
                {
                    if(string.IsNullOrEmpty(value))
                    {
                        throw new ArgumentException("Name cannot be blank","Name");
                    }
                    this.name = value;
                }
            }
            ....//其它属性
        }    
View Code

   

        这样是不是很方便?类似的,属性添加多线程支持更方便、可以作为接口定义的一部分、可以声明为虚属性、访问权限控制等。这些都是平常大家都会用的,此处就不再细说。
    2. 索引器
       如果类型接口需要包含一些索引数据项,那么这个时候可以使用索引器。C#中索引器又叫含参属性(parameterized property)。这个语法大家一般不会声明,但却都在使用。比如说
        List<int> intList = new List<int>(); 
        Console.Write(intList[0]);
       大家可能经常写这种代码,但只是对这个语法的名字不太熟悉,或者经常写但不知道如何声明这种语法。其实这个就是索引器,声明的语法也很简单。比如一个人可能有多个地址(工作地址、家庭住址、籍贯地址等等),代码如下:

    public class Person
            {
                private Dictionary<string,Address> addrDic = new Dictionary<string,string>();
    public Address this[string name]
                {
                    get
                    {
                        return addrDic[name];
                    }
                    set
                    {
                        this.addrDic[name] = value;//此处也可以改为先判断有没有key,没有key就添加。
                    }
                }
            }

      除此之外,C#还支持多维索引器,比如public object this[row,col],可以表示一个excel表格的某行某列的单元格里面的值,这样封装后的Excel帮助类肯定比xx.GetExcelValue(row,col)这种方法更容易使用。

属性有啥不好?     

显而易见,使用属性的代码没有使用数据成员的代码效率快(因为属性本质上是方法)。但是,属性也不见得比说那个数据成员的代码慢。这是为什么呢?JIT编译器会对方法进行内联处理(具体含义自行google),属性作为方法的一种也是会进行内联处理,那么属性和数据成员的效率就没有区别。当然,即便没有被内联,属性调用的效率相对于函数调用的成本也是可以忽略不计的,只有在一些极少数的情况下这种差别才值得我们注意。

可不可以先写公有字段,然后再改为属性?     

这个问题,大家只记住不可以就行了。具体原因如下:
    1. 对于属性和公有字段使用的源代码看起来一样但IL代码是不一样的;
    2. 如果一个类型的公有数据成员改为属性,那么会破坏二进制的兼容性,在程序已经部署的情况下可能会带来升级的麻烦。

总结

1. 对于暴漏在类型的公有接口或者受保护接口中的数据,我们应该使用属性。
2. 对于具有序列或者字典特征的类型,我们应该采用索引器。
3. 对于所有的数据成员,我们应该都应该声明为私有。

Effective c#学习笔记(1)

上一篇:C#结构体


下一篇:laravel 通过migration改变字段类型或枚举增加种类