Effective c#学习笔记(1)

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

属性是个啥,谁在用?

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

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

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

   public class Person
        {
            private string name;
            public string Name
            {
                get{return name;}
                set{this.name = value;}
            }
            ....//其它属性
        }     

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

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

   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;
                }
            }
            ....//其它属性
        }    

这样是不是很方便?类似的,属性添加多线程支持更方便、可以作为接口定义的一部分、可以声明为虚属性、访问权限控制等。这些都是平常大家都会用的,此处就不再细说。
    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. 对于所有的数据成员,我们应该都应该声明为私有。

时间: 2024-12-19 19:57:17

Effective c#学习笔记(1)的相关文章

原型模式和Effective C++学习笔记

原型模式(Prototype):用原型实例制定创建对象的种类,并且听过拷贝这些原型创建新的对象. 浅复制:如果字段是值类型的,则对该字段执行逐位复制,如果字段是引用类型,则复制引用但不复制引用的对象:因此,原始对象及其副本引用同一对象. 深复制:把引用变量的对象指向复制过的新对象,而不是原有的被引用的对象. Effective C++: 1:在资源管理类中提供对原始资源的访问. (1),APIs 往往要求访问原始资源(raw resources),所以每一个 RAII class 应该提供一个"

模板方法模式和Effective C++学习笔记

模板方法模式: 定义:定义一个操作中的算法的骨架,而将一些步骤延伸到子类中.模板方法使得子类可以不改变算法的结构即可重定义该算法的某些特定步骤. (1),用了继承,并且肯定这个继承有意义的情况下,就应该要成为子类的模板,所以重复的代码都应该提升到父类中,而不是让每个子类去重复. (2),当我们要完成某一个细节层次一致的过程或者一系列步骤,但其个别步骤在更详细的层次上的实现可能不同时,我们通常考虑模板方法模式来处理. (3),模板方法模式通过把不变的行为搬移到超类,去除子类中的重复代码来体现它的优

Effective C++学习笔记(Part Seven:Item 41-48)

 最近终于把effectvie C++仔细的阅读了一边,很惊叹C++的威力与魅力.最近会把最近的读书心得与读书笔记记于此,必备查找使用,如果总结有什么不当之处,欢迎批评指正: 现在只列出框架,最近会尽快填充完整: 第7部分:模板与泛型编程 所谓泛型编程就是以独立于任何特定类型的方式编写代码.使用泛型程序时,我们需要提供具体程序实例所操作的类型或值.模板是泛型编程思想的一种实现,也是C++研发者思想精髓所在. 条款41:了解隐式接口和编译器多态 条款42:了解typename的双重含义 条款4

Effective C++学习笔记(Part Six:Item 32-40)

 最近终于把effectvie C++仔细的阅读了一边,很惊叹C++的威力与魅力.最近会把最近的读书心得与读书笔记记于此,必备查找使用,如果总结有什么不当之处,欢迎批评指正: 现在只列出框架,最近会尽快填充完整: 第6部分:继承与面向对象设计 面向对象语言有三大特性:继承,封装和多态.遥想当年,每次校招笔试的时候都会有的,但能够甚至这六个字内涵的人,不再多数.本部分概括的说,深入的讲解三大特性之一----继承性.本部分花的时间不少,也感觉挺有意思的.特别是针对某个条款编写个具体程序实现后,明

Effective Java 学习笔记之第七条——避免使用终结(finalizer)方法

避免使用终结方法(finalizer) 终结方法(finalizer)通常是不可预测的,也是很危险的,一般情况下是不必要的. 不要把finalizer当成C++中析构函数的对应物.java中,当对象不可达时(即没有引用指向这个对象时),会由垃圾回收器来回收与该对象相关联的内存资源:而其他的内存资源,则一般由try-finally代码块来完成类似的工作. 一.finalizer的缺点: 1. 终结方法的缺点在于不能保证会被及时地执行. 及时执行finalizer方法是JVM垃圾回收方法的一个主要功

Effective C++学习笔记(Part Two:Item 5-12)

?? 近期最终把effectvie C++细致的阅读了一边,非常惊叹C++的威力与魅力.近期会把近期的读书心得与读书笔记记于此.必备查找使用,假设总结有什么不当之处,欢迎批评指正: 如今仅仅列出框架.近期会尽快填充完整: 第2部分:构造/析构/赋值运算 个人觉得这是C++的比較难理解的地方,也是C++的深邃所在,刚開始阅读时,可能在某些地方不理解,最好的方式是边读边编程实现,用心去思考其精髓所在! 条款5:了解C++默默编写并调用那些函数 条款6:若不想使用编译器自己主动生成的函数,就该明白拒绝

Effective C++学习笔记之对RAII思想的思考

 在阅读  Effective C++时,对RAII开始不理解,现总结如下: RAII是Resource acquisition is initialization的缩写形式,即"资源获取就是初始化",是C++等编程语言常用的管理资源.避免内存泄露的方法.它保证在任何情况下,使用对象时先构造对象,最后析构对象. 根据RAII 对资源的所有权可分为常性类型和变性类型,代表者分别是boost::shared_ptr和std::auto_ptr:从所管资源的初始化位置上可分为外部初始化类

Effective C++学习笔记(Part Three:Item 13-17)

 最近终于把effectvie C++仔细的阅读了一边,很惊叹C++的威力与魅力.最近会把最近的读书心得与读书笔记记于此,必备查找使用,如果总结有什么不当之处,欢迎批评指正: 现在只列出框架,最近会尽快填充完整: 第3部分:资源管理(Resource Management) 个人认为这是C++的比较关键的地方,利用好,可以使项目节约资源,合理利用内存资源,防止内存泄露.对于"资源"这两个字,可能刚接触,比较抽象,但你结合一下实际,也有很具体了! 条款13:以对象管理资源 条款14:

Effective C++学习笔记(Part Four:Item 18-25)

 最近终于把effectvie C++仔细的阅读了一边,很惊叹C++的威力与魅力.最近会把最近的读书心得与读书笔记记于此,必备查找使用,如果总结有什么不 当之处,欢迎批评指正: 现在只列出框架,最近会尽快填充完整: 第4部分:设计与申明(Designs and Declarations) 本部分主要讲述了类的设计原则以及设计规则,以及成员命令.成员函数设计时,应该注意的一些地方:个人感觉这是设计好一个类必不可少的. 条款18:让接口容易本正确使用,不易被误用 条款19:设计class犹如设计

Effective C++学习笔记(Part Five:Item 26-31)

 最近终于把effectvie C++仔细的阅读了一边,很惊叹C++的威力与魅力.最近会把最近的读书心得与读书笔记记于此,必备查找使用,如果总结有什么不 当之处,欢迎批评指正: 现在只列出框架,最近会尽快填充完整: 第5部分:实现(Implementations) 本部分主要讲述了类实现过程中的一些原则,为设计出高效率的实现方式,这是你的必经之路:在这一部分读的比较仔细,因为本人在类型转换这方面吃了不少苦头. 条款26:尽可能延后变量定义式的出现时间 条款27:尽量少做转型动作 条款28:避