声明:本文最初是本人从他出转载到51CTO上的一篇文章,但现在记不清最初从出处了,原文作者看到还请原来,现在发表在这里只为学习,本人在51CTO的该文章的地址为:http://kestrelsaga.blog.51cto.com/3015222/751536
一、 new运算符
① 用于创建对象和调用构造函数。 例如:
- Class1 obj = new Class1();
② 还可用于创建匿名类型的实例:
- var query = from cust in customers
- select new {Name = cust.Name, Address = cust.PrimaryAddress};
③ new 运算符还用于调用值类型的默认构造函数。 例如:
- int i = new int();
在上一个语句中,i 初始化为 0,它是 int 类型的默认值。 该语句的效果等同于:
- int i = 0;
请记住,为结构声明默认的构造函数是错误的,因为每一个值类型都隐式具有一个公共的默认构造函数。 可以在结构类型上声明参数化构造函数以设置其初始值,但是,只有在需要默认值之外的值时才必须这样做。
值类型对象(例如结构)是在堆栈上创建的,而引用类型对象(例如类)是在堆上创建的。 两种类型的对象都是自动销毁的,但是,基于值类型的对象是在超出范围时销毁,而基于引用类型的对象则是在对该对象的最后一个引用被移除之后在某个不确定的时间销毁。 对于占用固定资源(例如大量内存、文件句柄或网络连接)的引用类型,有时需要使用确定性终止以确保对象被尽快销毁。
不能重载 new 运算符。
如果 new 运算符分配内存失败,将引发异常 OutOfMemoryException。
示例
在下面的示例中,通过使用 new 运算符创建并初始化一个 struct 对象和一个类对象,然后为它们赋值。 显示了默认值和所赋的值。
- struct SampleStruct
- {
- public int x;
- public int y;
- public SampleStruct(int x, int y)
- {
- this.x = x;
- this.y = y;
- }
- }
- class SampleClass
- {
- public string name;
- public int id;
- public SampleClass() {}
- public SampleClass(int id, string name)
- {
- this.id = id;
- this.name = name;
- }
- }
- class ProgramClass
- {
- static void Main()
- {
- // Create objects using default constructors:
- SampleStruct Location1 = new SampleStruct();
- SampleClass Employee1 = new SampleClass();
- // Display values:
- Console.WriteLine("Default values:");
- Console.WriteLine(" Struct members: {0}, {1}",
- Location1.x, Location1.y);
- Console.WriteLine(" Class members: {0}, {1}",
- Employee1.name, Employee1.id);
- // Create objects using parameterized constructors:
- SampleStruct Location2 = new SampleStruct(10, 20);
- SampleClass Employee2 = new SampleClass(1234, "Cristina Potra");
- // Display values:
- Console.WriteLine("Assigned values:");
- Console.WriteLine(" Struct members: {0}, {1}",
- Location2.x, Location2.y);
- Console.WriteLine(" Class members: {0}, {1}",
- Employee2.name, Employee2.id);
- }
- }
- /*
- Output:
- Default values:
- Struct members: 0, 0
- Class members: , 0
- Assigned values:
- Struct members: 10, 20
- Class members: Cristina Potra, 1234
- */
注意,在示例中字符串的默认值为 null, 因此未显示它。
二、 new修饰符
在用作修饰符时,new 关键字可以显式隐藏从基类继承的成员。 隐藏继承的成员时,该成员的派生版本将替换基类版本。 虽然可以在不使用 new 修饰符的情况下隐藏成员,但会生成警告。 如果使用 new 显式隐藏成员,则会取消此警告,并记录要替换为派生版本这一事实。
若要隐藏继承的成员,请使用相同名称在派生类中声明该成员,并使用 new 修饰符修饰该成员。 例如:
- public class BaseC
- {
- public int x;
- public void Invoke() { }
- }
- public class DerivedC : BaseC
- {
- new public void Invoke() { }
- }
在此示例中,DerivedC.Invoke 隐藏了 BaseC.Invoke。 字段 x 不受影响,因为它没有被类似名称的字段隐藏。
通过继承隐藏名称采用下列形式之一:
· 引入类或结构中的常数、指定、属性或类型隐藏具有相同名称的所有基类成员。
· 引入类或结构中的方法隐藏基类中具有相同名称的属性、字段和类型。 同时也隐藏具有相同签名的所有基类方法。
· 引入类或结构中的索引器将隐藏具有相同名称的所有基类索引器。
对同一成员同时使用 new 和 override 是错误的做法,因为这两个修饰符的含义互斥。 new 修饰符会用同样的名称创建一个新成员并使原始成员变为隐藏的。 override 修饰符会扩展继承成员的实现。
在不隐藏继承成员的声明中使用 new 修饰符将会生成警告。
示例
在该例中,基类 BaseC 和派生类 DerivedC 使用相同的字段名 x,从而隐藏了继承字段的值。 该示例演示了 new 修饰符的用法。 另外还演示了如何使用完全限定名访问基类的隐藏成员。见:
- public class BaseC
- {
- public static int x = 55;
- public static int y = 22;
- }
- public class DerivedC : BaseC
- {
- // Hide field ‘x‘.
- new public static int x = 100;
- static void Main()
- {
- // Display the new value of x:
- Console.WriteLine(x);
- // Display the hidden value of x:
- Console.WriteLine(BaseC.x);
- // Display the unhidden member y:
- Console.WriteLine(y);
- }
- }
- /*
- Output:
- 100
- 55
- 22
- */
在此示例中,嵌套类隐藏了基类中同名的类。 此示例演示了如何使用 new 修饰符来消除警告消息,以及如何使用完全限定名来访问隐藏的类成员。见:
- public class BaseC
- {
- public class NestedC
- {
- public int x = 200;
- public int y;
- }
- }
- public class DerivedC : BaseC
- {
- // Nested type hiding the base type members.
- new public class NestedC
- {
- public int x = 100;
- public int y;
- public int z;
- }
- static void Main()
- {
- // Creating an object from the overlapping class:
- NestedC c1 = new NestedC();
- // Creating an object from the hidden class:
- BaseC.NestedC c2 = new BaseC.NestedC();
- Console.WriteLine(c1.x);
- Console.WriteLine(c2.x);
- }
- }
- /*
- Output:
- 100
- 200
- */
如果移除 new 修饰符,该程序仍可编译和运行,但您会收到以下警告:The keyword new is required on ‘MyDerivedC.x‘ because it hides inherited member ‘MyBaseC.x‘.
三、 new约束
new 约束指定泛型类声明中的任何类型参数都必须有公共的无参数构造函数。 如果要使用 new 约束,则该类型不能为抽象类型。
示例
当泛型类创建类型的新实例,请将 new 约束应用于类型参数,如下面的示例所示:
- class ItemFactory<T> where T : new()
- {
- public T GetNewItem()
- {
- return new T();
- }
- }
当与其他约束一起使用时,new() 约束必须最后指定:
- public class ItemFactory2<T>
- where T : IComparable, new()
- {
- }