C#锐利体验(五)

2016-02-19 17:08 5 1 收藏

下面图老师小编要跟大家分享C#锐利体验(五),简单的过程中其实暗藏玄机,还是要细心学习,喜欢还请记得收藏哦!

【 tulaoshi.com - 编程语言 】

  第五讲 构造器与析构器

  构造器

  构造器负责类中成员变量(域)的初始化。C#的类有两种构造器:实例构造器和静态构造器。实例构造器负责初始化类中的实例变量,它只有在用户用new关键字为对象分配内存时才被调用。而且作为引用类型的类,其实例化后的对象必然是分配在托管堆(Managed Heap)上。这里的托管的意思是指该内存受.NET的CLR运行时管理。和C++不同的是,C#中的对象不可以分配在栈中,用户只声明对象是不会产生构造器调用的。

  实例构造器分为缺省构造器和非缺省构造器。缺省构造器是在一个类没有声明任何构造器的情况下,编译器强制为该类添加的一个无参数的构造器,该构造器仅仅调用父类的无参数构造器。缺省构造器实际上是C#编译器为保证每一个类都有至少一个构造器而采取的附加规则。注意这里的三个要点:

  子类没有声明任何构造器;

  编译器为子类加的缺省构造器一定为无参数的构造器;

  父类一定要存在一个无参数的构造器。

  看下面例子的输出:

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/bianchengyuyan/)
using System;public class MyClass1{    public MyClass1()    {        Console.WriteLine(MyClass1Parameterless Contructor!);    }    public MyClass1(string param1)    {        Console.WriteLine(MyClass1Constructor Parameters : +param1);    }}public class MyClass2:MyClass1{}public class Test{    public static void Main()    {        MyClass2 myobject1=new MyClass2();    }}

  编译程序并运行可以得到下面的输出:

MyClass1 Parameterless Contructor! 

  读者可以去掉MyClass1的无参构造器public MyClass1()看看编译结果。

  构造器在继承时需要特别的注意,为了保证父类成员变量的正确初始化,子类的任何构造器默认的都必须调用父类的某一构造器,具体调用哪个构造器要看构造器的初始化参数列表。如果没有初始化参数列表,那么子类的该构造器就调用父类的无参数构造器;如果有初始化参数列表,那么子类的该构造器就调用父类对应的参数构造器。看下面例子的输出:

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/bianchengyuyan/)
using System;public class MyClass1{    public MyClass1()    {        Console.WriteLine("MyClass1 Parameterless Contructor!");    }    public MyClass1(string param1)    {        Console.WriteLine("MyClass1Constructor Parameters : "+param1);    }}public class MyClass2:MyClass1{    public MyClass2(string param1):base(param1)    {        Console.WriteLine("MyClass2Constructor Parameters : "+param1);    }}public class Test{    public static void Main()    {        MyClass2 myobject1=new MyClass2("Hello");    }}

  编译程序并运行可以得到下面的输出:

MyClass1 Constructor Parameters : HelloMyClass2 Constructor Parameters : Hello 
C#支持变量的声明初始化。类内的成员变量声明初始化被编译器转换成赋值语句强加在类的每一个构造器的内部。那么初始化语句与调用父类构造器的语句的顺序是什么呢?看下面例子的输出:
using System;public class MyClass1{    public MyClass1()    {        Print();    }    public virtual void Print() {}}public class MyClass2: MyClass1{    int x = 1;    int y;    public MyClass2()    {        y = -1;        Print();    }    public override void Print()    {        Console.WriteLine("x = {0}, y = {1}", x, y);    }}public class Test{    static void Main()    {        MyClass2 MyObject1 = new MyClass2();    }}

  编译程序并运行可以得到下面的输出:

x = 1, y = 0x = 1, y = -1 

  容易看到初始化语句在父类构造器调用之前,最后执行的才是本构造器内的语句。也就是说变量初始化的优先权是最高的。

  我们看到类的构造器的声明中有public修饰符,那么当然也可以有protected/private/ internal修饰符。根据修饰符规则,我们如果将一个类的构造器修饰为private,那么我们在继承该类的时候,我们将不能对这个private的构造器进行调用,我们是否就不能对它进行继承了吗?正是这样。实际上这样的类在我们的类内的成员变量都是静态(static)时,而又不想让类的用户对它进行实例化,这时必须屏蔽编译器为我们暗中添加的构造器(编译器添加的构造器都为public),就很有必要作一个private的实例构造器了。protected/internal也有类似的用法。

  类的构造器没有返回值,这一点是不言自明的。

  静态构造器初始化类中的静态变量。静态构造器不象实例构造器那样在继承中被隐含调用,也不可以被用户直接调用。掌握静态构造器的要点是掌握它的执行时间。静态构造器的执行并不确定(编译器没有明确定义)。但有四个准则需要掌握:

  在一个程序的执行过程中,静态构造器最多只执行一次。

  静态构造器在类的静态成员初始化之后执行。或者讲编译器会将静态成员初始化语句转换成赋值语句放在静态构造器执行的最开始。

  静态构造器在任何类的静态成员被引用之前执行。

  静态构造器在任何类的实例变量被分配之前执行。

  看下面例子的输出:

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/bianchengyuyan/)
using System;class MyClass1{    static MyClass1()    {        Console.WriteLine("MyClass1 Static Contructor");    }    public static void Method1()    {        Console.WriteLine("MyClass1.Method1");    }}class MyClass2{    static MyClass2()    {        Console.WriteLine("MyClass2 Static Contructor");    }    public static void Method1()    {        Console.WriteLine("MyClass2.Method1");    }}class Test{    static void Main()    {        MyClass1.Method1();        MyClass2.Method1();    }}

  编译程序并运行可以得到下面的输出:

MyClass1 Static ContructorMyClass1.Method1MyClass2 Static ContructorMyClass2.Method1 

  当然也可能输出:

MyClass1 Static ContructorMyClass2 Static ContructorMyClass1.Method1MyClass2.Method1 

  值得指出的是实例构造器内可以引用实例变量,也可引用静态变量。而静态构造器内能引用静态变量。这在类与对象的语义下是很容易理解的。

  实际上如果我们能够深刻地把握类的构造器的唯一目的就是保证类内的成员变量能够得到正确的初始化,我们对各种C#中形形色色的构造器便有会心的理解--它没有理由不这样!

  析构器

  由于.NET平台的自动垃圾收集机制,C#语言中类的析构器不再如传统C++那么必要,析构器不再承担对象成员的内存释放--自动垃圾收集机制保证内存的回收。实际上C#中已根本没有delete操作!析构器只负责回收处理那些非系统的资源,比较典型的如:打开的文件,获取的窗口句柄,数据库连接,网络连接等等需要用户自己动手释放的非内存资源。我们看下面例子的输出:

  

using System;class MyClass1{    ~MyClass1()    {        Console.WriteLine("MyClass1's destructor");    }}class MyClass2: MyClass1{    ~MyClass2()    {        Console.WriteLine("MyClass2's destructor");    }}public class Test{    public static void Main()    {        MyClass2 MyObject = new MyClass2();        MyObject = null;        GC.Collect();        GC.WaitForPendingFinalizers();    }}

来源:http://www.tulaoshi.com/n/20160219/1614317.html

延伸阅读
第五章:代理 代理实现的是象c++等语言的指针功能,不同于函数指针,代理是一种面向对象、安全类型的。代理事派生于公共基类(system)的一种参考类型,方法被压入一个代理中,对于实例方法被称为实例的组成实体或关于实例的方法,而静态方法,被称为类的组成实体或类方法。代理的强大功能是它可以自动的匹配方法,而不管其类型。 写一个代理...
标签: ASP
       欢迎您加入C#的世界!      这一章将把您引进C#的天地,并回答一些相关的问题,如:您为什么要使用C#,C++和C#的主要有什么不同点,以及为什么C#使开发更容易而且还使您感到很有趣。      为什么是另外一种编程语言?      必须回答...
应该说,从 C/C++ 发展到 C#,进而到 Microsoft .NET Framework,对于习惯使用 C/C++ 的程序员来说是一次小小的打击。换句话说,C# 编程将带来翻天覆地的变化,不仅要完成范例转换,还要完成程序模型的重建以及其他各种令人难以捉摸的词组转变。但是,随着新一年的到来,不正是学习新编程语言的好时机吗? 我现在实在是抽不出时间教您如何使...
在过去的二十年里,C和C++已经成为在商业软件的开发领域中使用最广泛的语言。它们为程序员提供了十分灵活的操作,不过同时也牺牲了一定的效率。与诸如Microsoft? Visual Basic? 等语言相比,同等级别的C/C++应用程序往往需要更长时间来开发。由于C/C++语言的复杂性,许多程序员都试图寻找一种新的语言,希望能在功能与效率之间找到一个更...
   1 绪论 c# 是一种简练,时髦(?),面向对象(object oriented),类型可靠(type-safe)的 编程语言。它(发音:C sharp)是从c/c++发展而来的(?俺觉得更象是java),和c/c++ 是一个语系。所以,很容易被c/c++的程序员接受。c#的目标是结合Visual Basic的高产和 C++质朴的力量。 c#将会是vs7的一分子。vs7还支持vb,vc和...

经验教程

209

收藏

45
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部