最近面試中有一道題是寫new關鍵字的幾種用法,想了下寫下我知道的兩種用法
第一種 創建對象、調用構造函數,這就不用講了 ClassA A=new ClassA();
第二種 是作為修飾符,顯示隱藏繼承於基類的繼承成員
class Program { static void Main(string[] args) { Class1 cls1 = new Class1(); Class2 cls2 = new Class2(); cls1.Prinf(); cls2.Prinf(); Test t1 = new Class1(); Test t2 = new Class2(); t1.Prinf(); t2.Prinf(); Console.ReadLine(); } abstract public class Test { public virtual void Prinf() { Console.WriteLine("base Printf..."); } } public class Class1 : Test { public override void Prinf() { Console.WriteLine("Class1 Printf..."); } } public class Class2 : Test { public new void Prinf() { Console.WriteLine("Class2 Printf..."); } } }
我們可以看到 打印的最后一條信息 子類繼承於父類的方法被隱藏,實現了父類的方法,override 則完全重寫了方法,即使轉變為基類對象調用的也是派生類的重寫方法。
第三種 是用在泛型中添加類型的約束
class Test<T> where T : new()//定義類型T的約束,表示T類型必須有不帶參數的構造函數 { public T GetItem() { // return T(); return new T();//如果不添加new()約束,編譯錯誤:變量類型“T”沒有 new() 約束,因此無法創建該類型的實例 //想一下,T類型不知道,編譯器不知道分配多大的空間,所以會通過反射技術實現 } } class TClass { private int a; public TClass() //如果不添加無參構造函數,編譯錯誤:TClass必須是具有公共的無參數構造函數的非抽象類型,才能用作泛型類型或方法“A.Test<T>”中的參數“T” { } public TClass(int a) { this.a = a; } } class Program { static void Main(string[] args) { Test<TClass> test = new Test<TClass>(); Console.ReadLine(); }
}