轻松学习的有效方法

进修社 人气:1.78W

一、静态方法和实例方法

轻松学习的有效方法

静态方法是一个特殊的成员方法,不属于类的某一个具体的实例或对象,而属于类本身。静态方法不对特定实例进行操作,只能访问类中的静态成员。访问静态方法只能使用类名,而不需要创建对象,也不能使用对象名类引用,声明静态方法修饰符中必须有static关键字。

实例方法可以使用类的任何成员。调用实例方法时,必须使用类的实例或对象来引用。实例方法对类的某个给定的实例进行操作,在实例方法类中可以使用this来访问实例。调用实例方法时,必须先创建一个对象。

简单的说,静态方法只能访问静态成员,实例方法可以访问静态和实例成员。之所以不允许静态方法访问实例成员变量,是因为实例成员变量是属于某个对象的,而静态方法在执行时,并不一定存在对象。同样,因为实例方法可以访问实例成员变量,如果允许静态方法调用实例方法,将间接地允许静态方法使用实例成员变量,这是错误的。基于同样的道理,静态方法中不能使用关键字this。

例一、编程使用静态方法和实例方法

using System; using ric; using ; using ; using s; namespace Text { class A { int exaVar;//创建的一个为实例成员变量 static int stVar;//创建的一个静态成员变量 void tM()//实例方法 { exaVar = 1;//等价于r=1 stVar = 1;//等价于r=1 } static void sM()//静态方法 { //exaVar = 1;//错误,静态方法不可以调用实例成员变量 stVar = 1;//等价于r=1 } static void Main(string[] args) { A text = new A();//创建类A的对象为text ar = 1;//对象text访问实例成员变量 r = 1;//只能使用类访问静态成员变量 //r = 1;//不能使用对象text访问静态成员变量 ();//使用对象text访问实例成员方法 //();//不能使用对象text访问静态成员方法 ();//使用类访问静态成员方法 eLine(ar + r); Line(); } } }

输出的结果为:2

二、虚方法和非虚方法

若一个实例方法的声明中含有virtual修饰符,则称该方法为虚方法。若其中没有virtual修饰符,则称该方法为非虚方法。

非虚方法的实现是一成不变的,无论该方法是在声明它的类的实例上调用还是在派生类的实例上调用,实现均相同。与此相反,虚方法的实现可以由派生类取代。取代所继承的虚方法的实现的过程称为重写该方法。在虚方法调用中,该调用所涉及的那个实例运行时类型确定了要被调用的究竟是该方法的哪一种实现。在非虚方法调用中,相关的实例的编译时类型是决定性因素。

例二、使用虚方法和非虚方法在派生类中调用

using System; using ric; using ; using ; using s; namespace Text { public class A { public virtual void ab()//定义的类A的虚方法 [csharp] view plaincopyprint? { eLine("这是虚方法"); } public void ac()//定义的类A的非虚方法 { eLine("这是非虚方法"); } } public class B:A//类B私有继承类A { public override void ab()//重写继承的ab方法 { eLine("这是新的方法"); } public new void ac()//创建一个新的方法覆盖原来的ac方法 { eLine("这是另一个新的方法"); } } class Program { static void Main(string[] args) { B b = new B();//创建类B的对象b A a = b;//将对象b赋值给类A的对象a ();//调用原ab虚方法 ();//调用ab方法 ();//调用原ac非虚方法 ();//调用ac方法 Line(); } } }

从输出的结果中可以看出虚方法的实现是由派生类取代并且由它的方法实现,而非虚方法的实现是定义的类和派生类的各自由它们的方法实现。

三、虚方法和重写方法

重写方法用相同的签名重写所继承的虚方法。虚方法声明用于引入新方法,而重写方法声明则用于使现有的继承虚方法专用化。用override声明所重写的那个方法称为已重写了的基方法。

重写声明和已重写了的基方法具有相同的声明可访问性。换句话说,重写声明不能更改所对应的 虚方法的可访问性。但是,如果已重写的基方法是protected,并且声明它的程序集不是包含重写方法的程序集,则重写方法声明的可访问性必须是protected。

重写override一般用于接口的实现和继承类的方法改写时应注意:

(1)覆盖的方法的标志要和被覆盖的方法的标志完全匹配,才能达到覆盖的效果

(2)覆盖的方法的返回值必须和被覆盖的方法的返回值一致

(3)覆盖的方法所抛出的异常必须和被覆盖的方法所抛出的异常一致,或者是其子类

(4)被覆盖的方法不能为private,否则在其子类中只是新定义了一个方法,并没有对其进行覆盖

例三、创建一个虚方法并重写这个虚方法,通过调用两个方法比较结果

using System; using ric; using ; using ; using s; namespace Text { class A { public virtual void a()//创建的虚方法 { eLine("这是虚方法"); } } class B : A { public override void a() { eLine("这是重写后的`方法"); } } class Program { static void Main(string[] args) { B b1 = new B();//创建类B的对象b1 b1.a();//调用重写后的a方法 A a1 = new A();//创建类A的对象a1 a1.a();//调用虚方法a A a2 = b1;//将类B的对象b1赋值给类A的对象a2 a2.a();//调用虚方法a Line(); } } }

输出的结果为:这是重写后的方法

这是虚方法

这是重写后的方法

四、外部方法(这个方法很新奇,联想到很多)

当方法声明包含extern修饰符时,称该方法为外部方法。外部方法是在外部实现的,编程语言通常是使用C#以外的语言。外部方法不可以是泛型。

extern修饰符通常与DllImport属性一起使用,从而使外部方法可以由DLL(动态链接库)实现。执行环境可以支持其他用来提供外部方法实现的机制。当外部方法包含DllImport属性时,该方法声明必须同时包含一个static修饰符。

在使用DLLImport属性时,一定要引入ropServices命名空间,此命名空间是提供各种各样支持COM interop及平台调用服务的成员。

例四、通过外部方法和Dlllmport属性调用“”实例自定义信息提示框的功能

using System; using ric; using ; using ; using s; using ropServices;//必须引入的命名空间 namespace Text { class Program { [DllImport("")]//调用 public static extern int MessageBox(int H, string m, string c, int type);//定义的外部方法 static int Main(string[] args) { e("请输入信息:"); string str = Line();//接受输入的信息 return MessageBox(0,str,"我的信息提示框",0);//以信息提示框输出 } } }

五、分部方法

若一个方法声明中含有partial修饰符,则称该方法为分部方法。只能讲分部方法声明为分部类型的成员,而且要遵守约束数目。分部方法有着严格的限制。分部方法必须在分部类或分部机构内声明。它们必须是私有的,不能返回值,不能有输出参数。因为任何针对没有被实现的分部方法的调用都会简单地被忽略,所以说这些限制是非常有必要的。

分部方法不能作为一个明确分配的变量,仅被代码生成器在处理轻量级事件的时候使用。假设用户解析一个数据库或者一个XML文件,然后生成了数据类,结果会发现有数十个类,几百个属性以及一大堆泛型和模板文件等。分部方法另外一个经常被用到的地方是验证,或者让属性的setter去更新另一个属性。所以如果用户要使用产生的代码,或者在运行时有几百个事件和千个方法调用的话(其实大多数情况下只用到了其中的一点点),就可以选择分部方法。分部方法在声明和使用时要比事件容易得多,如果没有用到它们,它们就会消失。从分部方法必须是私有的限制中,

Alexander发现了该方法的不足之处,即如果用户喜欢原数据驱动的应用,并且已经被的数据绑定所困扰时(因为没有其他的方法可以附上原数据),那么,将来会出现丢失信息的可能。

例五、创建一个类的分部方法并且在另一个部分中引用

using System; using ric; using ; using ; using s; namespace Text { public partial class Program//定义的分部类 { private int _setup;//定义的字段 public int setup//定义的方法 { set { _setup=value; } get { return _setup; } } partial void text(int t);//声明分部方法 partial void text(int t)//实现声明 { t += _setup; e(t); } } partial class Program { static void Main(string[] args) { Program pg = new Program();//创建类对象 pg._setup = 100;//设置属性 (50);//调用方法 Line(); } } }

输出的结果为:150

TAGS:学习