程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> .NET網頁編程 >> C# >> C#基礎知識 >> 《Effective C# 精髓》摘選

《Effective C# 精髓》摘選

編輯:C#基礎知識

  昨天買了一本《Effective C#》,看了幾個Item,雖然沒有當初讀《Effective C++》時的那般震撼,但是也收獲不少。把其中的要點記錄於下,有些條款加上了自己的理解,權當作讀書筆記吧 :-)

  Item 1: Always Use Properties Instead of Accessible Data Members

  這個是地球人都知道的條款了。你需要記住,屬性是類的外部接口部分,而(公共)成員卻是內部實現。如果把內部實現暴露給外部,對於以後類的實現變更是非常不利的。

  Item 2: Prefer readonly to const

  這個條款需要注意一下幾點:

  (1)const在編譯期發生作用,即編譯器會將所有的const成員置換成對應的常量“值”。

  (2)即使引用其他程序集中的const成員,本程序集中也是硬編碼了const成員的值。

  (3)readonly在運行期被評估,所以其性能比const稍差,但是靈活性更高。

  (4)const的值必須在編譯期決定,所以不能使用new為其賦值。

  (5)更新一個公有的const成員的值應被視為接口改變,而更新一個readonly變量的值可視為內部實現的改變。

  Item 3: Prefer the is or as Operators to Casts

  (1)is或as稱為“動態轉換”,是嘗試性的,如果失敗,不會拋出異常。盡可能使用as操作符。該機制使用元數據完成功能。

  (2)Cast稱為“強制轉換”,如果失敗,則拋出異常--代價高昂。

  (3)is、as、Cast轉換都不會調用自定義的轉換操作符。

  (4)is可以判斷一個object是否為值類型,而as不行。

  (5)請注意Type.IsAssignableFrom()和Type.IsSubclassOf()方法,他們也是常用的“類型檢測”手段。注意,Type.IsSubclassOf()方法不支持接口檢測,而Type.IsAssignableFrom()支持。

  Item 4: Use Conditional Attributes Instead of #if

  使用#if常(可能)導致性能問題(如空方法調用)和程序對#if/#endif塊代碼的依賴問題。

  (1)使用Conditional Attributes修飾的方法總是會被編譯到目標程序集中,無論是Release或Debug。

  (2)如果條件不滿足該Conditional Attributes指定的條件,則編譯器會忽略所有對其修飾的方法的調用。

  (3)被Conditional Attributes修飾的方法必須返回void,這是有道理的。因為我們的程序運行不能依賴被Conditional Attributes修飾的方法的返回值。否則,在不同的條件下,我們的程序將表現出非我們期望的不用行為。

  Item 5: Always Provide ToString()

  關於這一點,我在以往的項目中早有體會。舉個例子,曾經我們需要把從數據庫中取出的Customer列表綁定到ComboBox,開始時我們設計Customer時並沒有重寫ToString()方法,所以我們要這樣做:

            //從數據庫中挑出所有有效用戶
            string whereStr = string.Format("where {0} = '1'" ,Customer._IsValid) ;
            Customer[] customers = (Customer[])DataEntrance.GetObjects(typeof(Customer) ,whereStr) ;
            ArrayList cusNameList = new ArrayList() ;
            foreach(Customer cus in customers)
            {
                cusNameList.Add(string.Format("{0} {1}" ,cus.ID ,cus.Name)) ;
            }
            //綁定
            this.comboBox1.DataSource = cusNameList ;

  如果為Customer重寫ToString()方法,

        #region ToString 
        public override string ToString()
        {
            return this.ID.ToString() + " " + this.Name.ToString() ;
        }
        #endregion

  則只需要這樣:

            string whereStr = string.Format("where {0} = '1'" ,Customer._IsValid) ;
            Customer[] customers = (Customer[])DataEntrance.GetObjects(typeof(Customer) ,whereStr) ;
            this.comboBox1.DataSource = customers ;

  這樣就簡便了好多,而且這樣做還有一個好處,比如,從ComboBox從選取一個客戶時,以前需要這樣:

            string cusID = this.comboBox1.SelectedItem.ToString().Split(' ')[0] ;
            Customer desCus = null ;
            foreach(Customer cus in customers)
            {
                if(cus.ID = cusID)
                {
                    desCus = cus ;
                    break ;
                }
            }
  現在,簡單多了,一行代碼搞定。

Customer desCus = this.comboBox1.SelectedItem as Customer ;

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved