c#中override virtual static abstract sealed 的作用

来源:互联网 发布:淘宝网白酒 编辑:程序博客网 时间:2024/04/28 09:35
 

说明1:

表示静态的关键字
说明此对象在应用中只存在一份 

说明2:

C# 是面向对象的程序设计语言,每一个函数都属于一个类。

当一个方法被声明为Static时,这个方法是一个静态方法,编译器会在编译时保留这个方法的实现。也就是说,这个方法属于类,但是不属于任何成员,不管这个类的实例是否存在,它们都会存在。就像入口函数Static void Main,因为它是静态函数,所以可以直接被调用。

当一个方法被声明为Virtual时,它是一个虚拟方法,直到你使用ClassName variable = new ClassName();声明一个类的实例之前,它都不存在于真实的内存空间中。这个关键字在类的继承中非常常用,用来提供类方法的多态性支持。

例如,有一个类Test,Test类拥有两个方法Hello()和greet():
public class test
{
public void Hello()
{
System.Console.WriteLine("hello, world!");
}
.....
}
类TestMe派生自类Test,那么当你使用如下的代码:
Test a = new TestMe();
创建一个TestMe类的新实例后,假如你试图执行如下的代码:
a.Hello();
那么,自然就会运行基类Test的Hello()方法,但是如果你想给予派生类TestMe一个它自己的Hello()方法,你就要在Test类中将Hello()方法声明为虚拟的:
public virtual void Hello()
{
...
}
然后在派生类中以 override关键字表示覆盖基类的方法:
public class TestMe : Test
{
...

public overrice void Hello()
{
System.Console.WriteLine("hello from TestMe class!!!");
}

....
}
这时调用a.Hello(),就会出现“hello from TestMe class!!!”字样,而不是“Hello,world!”说明基类的方法已经被覆盖了。这就是多态性的表现。

从上面不难看出,一个静态的方法是真实存在的,而一个虚拟方法可以被派生类重写,这二者是冲突的,其实对于一个方法,C#规定只能使用下面这些限定符中的一个:
override virtual static abstract sealed
代表的含义分别为:
重载函 数、虚拟函数、静态函数、抽象函数、密封函数(不可派生)

另外,C#中定义一个方法的声明为:

可见性 类型 返回值 方法名(参数列表){方法体}

例如
public static void Test(int a){System.Console.WriteLine(a.ToString());}

这是一个公有的静态函数,函数名为Test,无返回值,有一个整形参数a,作用是把a的值输出在屏幕上。

 

说明3:

比如有类
class a
{
static void function1()
{}
}
和类
class b
{
b()
{}
void function2()
{}
}
如果要使用function1的时候,直接使用类名a.function1()就可以了,如果要使用function2()的话需要对b进行构建一个对象 也就是:b haha = new b(),然后haha.function2(),可以这样进行使用,
也就是说对于static修饰过的方法或者属性,对于所有的该类的对象都只有相同的一个,而没有修饰过的则表示,对于不同的对象,会有不同的属性,比如不同的人有不同的名字,身高等。

说明4:

public static 最好一个人开车
public 可以很多人一起开车

 

 

 

===========================

 

 

 

new 与 override的区别

 引用AnyTao博客中的一句话:
覆写(override)与重载(overload),是成就.NET面向对象多态特性的基本技术之一
正 如某网友说的那一句话:
Override 覆写 就是把原来的换掉了
new        新的 就是两个共存着

使用override重写xx方法,通过父类引用一样只能看到重写后的方法;
如 果使用new隐藏xx方法,父类子类引用各自对应方法;

override重写虚方法,那么就只剩下重写以后的方法;
new隐藏基 类的方法,那么基类的方法和当前类的方法同时存在只是被隐藏了;

 

 

使用override修饰符主要用来修改方法、属性、索引器或事件。重写基方法必须与重写方法具有相同的名称。
        不能重写非虚方法或静态方法。重写基方法必须是虚拟的、抽象的或重写的。
       重写声明不能更改虚方法的可访问性。重写方法和虚方法必须具有相同的访问级修饰符。例如:虚方法为public的,重新方法也必须是public的。
        不能使用下列修饰符修改重写方法:
        new        static        virtual        abstract

        重写属性声明必须指写与继承属性完全相同的访问修饰符、类型和名称,并且重写属性必须是虚拟的、抽象的或是重写的。

要求:(三相同)
1、方法名称相同
2、参数列表相同
3、 返回值类型相同
一句话,只需要重新写方法内部的内容!

,override可以覆盖基类的方法,让基类的方法 以子类的内容实现,而new不用来覆盖基类的方法,而是全新定义一个子类的方法,这个方法只属于子类,与基类的方法无关,只是名字上相同而已

 


 

先看abstract和override使用方法

abstract class Base

    {

       publicvirtual void work()

        {

           MessageBox.Show("基类--开始工作");

        }

       publicvirtual void outwork()

        {

           MessageBox.Show("基类--下班");

        }

        

        Public abstract void Pay();      //声明抽象方法,必须要被子类new 或 override;只有当类是abstract时才可以声明abstract方法    

    }

   classEmployee : Base

    {

       publicnew void work()

        {

           MessageBox.Show("子类(new)--开始工作");

        }

       publicoverride void outwork()   //覆写抽象方法(一定要在子类覆写父类的抽象方法)

        {

            MessageBox.Show("子类(override)下班");

        }

              

//测试代码

        

c中override virtual static abstract sealed 的作用及代码分析 - 蒾亽迗哋 - 、Welcomes Space|゛c中override virtual static abstract sealed 的作用及代码分析 - 蒾亽迗哋 - 、Welcomes Space|゛Code
          //第一种情况
            Employee emp = new Employee();

            emp.work();              
//子类(new)--开始工作


            emp.outwork();          
//子类(override)下班     

            
//第二种情况

            Employee emp = new Employee();

            Base 
= (Base)emp;   
//指向抽象类了

            b.work();                
//基类--开始工作

            b.outwork();             
//子类(override)下班   被子类覆写了,所以不是显示基类下班

            
//第三种情况

            Base = new Employee();

            
//同上面是一样的


            b.work();                
//基类--开始工作

            b.outwork();             
//子类(override)下班

            Console.ReadLine();

   

new声明的方法,当使用子类的类型来调用的时候,它会运行子类中的函数,而如果类型是基类的话,被隐藏的基类函数就会站到前台来。只有使用virtual定义基类中的函数,并使用override标记子类中的函数,才可以达到想要的多态类(始终调用子类的方法)。

在子类中new父类中的方法,父类中的方法不一定是虚类型,即virtual

但是在子类中override父类中的方法,父类的方法一定得是虚类型,

 

代码示 例:

   abstractclass Animal

    {

       publicabstract void Drink();

       publicabstract void GotoBed();

    }

   classDog : Animal

    {

        publicoverridevoid Drink()

        {

           MessageBox.Show("小狗喝水");

        }

       publicoverride void GotoBed()

        {

           MessageBox.Show("小狗睡觉");

        }

       publicoverride string ToString()  //也可:public new string ToString()

        {

           return"小狗";

        }

    }

总结:

1、               抽象方法,必须要被子 override;只有当类是abstract时才可以声明abstract方法  

2、               因为abstract方法没有方法实现,其子类只能对其 abstract方法进行override,不能new(如果可以new的话,那么类的类型是父类的话,类的方法执行的是父类的方法而非子类的方法,而父类的方法却没有方法实现,那么将如何执行呢?)

3     Selaed方法必须与 override连用,也就是说实现sealed方法的类的父类必须实现了此方法(sealed关键字有两个作用:1,密封类不能被继承。2:密封方法重写基类中的方法,但其本身不能在任何派生类中进一步重写,Selaed方法必须与override连用

 

 

如下,A声明了virtual方法A1,那么 A的子类AA才能对Pay进行密封重写,AA的子类不能对A1重写或覆盖。

sealed方法的使用代码示例:

   classA

    {

       publicvirtual void A1()

        {

           MessageBox.Show("A---A1");   

        }

    }

   classAA:A

    {

       publicsealed override void A1()

        {

           MessageBox.Show("AA---A1");

              

    }

总结

Public abstract void pay();                 abstract方法没有方法实现,必须继承

Public sealed override void pay(){}      sealed方法必须是重写父类的方法

Public static void pay(){}                   static方法通过类名访问

Public virtual void pay(){}                  virtual方法子类可以对其overridenew

Public new void pay(){}                     父类的方法pay不一定是virtual

Public override void pay(){}               父类的方法pay一定要是virtual

 

 

 

new、abstract、virtual、 override关键字的使用代码示例

 

public abstract class People   //abstract 说明类People是一个抽象类,不能被实例的
    {
        
public
 People()
        {
        }

        
public void
 Work()
        {
            MessageBox.Show(
"开始工作!"
);
        }

        
public virtual void GetOffWork()   //虚函数,说明此方法可以被子类覆盖(override)

        {
            MessageBox.Show(
"下班啦!"
);
        
    

    
public class Manage:People   //继承Popele 类

    {
        
public
 Manage()
        {
        

        
new public void Work()   //因为基类已经实现了Work方法,而在子类中又实现了Work方法,                                         //所以编译器会报警,在前面加上 new(隐藏基类方法),是将警报关闭。

        {
            
base.Work();   //
调用基类Popele的方法。显示“开始工作”
            
//MessageBox.Show("管理员开始工作罗!");

        

        
public override void GetOffWork()      //覆盖基类的方法

        {
            MessageBox.Show(
"管理员下班啦"
);
        
    }

    
public class
 Employee():People
    {
       
public
 Employee()
       {}

        
new public void GetOffWork()      //virtual方法仍然可以 new,abstract方法不能用new

        {
            MessageBox.Show(
"职员下班啦!"
);
        

    }

以上应该应该 几点:
1
、如果父类方法没有加virtual关键字,即不是一个虚方法,则在子类中只能隐藏基类方法,而不能覆盖。
2
、如果父类方法加了virtual关键字,即它是一个虚方法,在子类中一样可以隐藏。
3
、如果子类实现了父类相同的方法(相同的方法名称及签名),而没有new,在编译时会报警,但编译仍然能够通过!
3、调用父类方法:base
.方法名()
4、abstract类 是一个抽象类,不能被实例化