c#中override virtual static abstract sealed 的作用

c#中override virtual static abstract sealed 的作用




 
说明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
 
    {
 
        public virtual void work()
 
        {
 
            MessageBox.Show("基类--开始工作");
 
        }
 
        public virtual void outwork()
 
        {
 
            MessageBox.Show("基类--下班");
 
        }
 
        
 
        Public abstract void Pay();       //声明抽象方法,必须要被子类new 或 override;只有当类是abstract时才可以声明abstract方法    
 
    }
 
    class Employee : Base
 
    {
 
        public new void work()
 
        {
 
            MessageBox.Show("子类(new)--开始工作");
 
        }
 
        public override void outwork()   //覆写抽象方法(一定要在子类覆写父类的抽象方法)
 
        {
 
            MessageBox.Show("子类(override)下班");
 
        }
 
}               
 
//测试代码
 
        
 
Code
          //第一种情况
            Employee emp = new Employee();

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

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

            //第二种情况
            Employee emp = new Employee();

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

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

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

            //第三种情况
            Base b = new Employee();

            //同上面是一样的

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

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

            Console.ReadLine();

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

    abstract class Animal
 
    {
 
        public abstract void Drink();
 
        public abstract void GotoBed();
 
    }
 
    class Dog : Animal
 
    {
 
        public override void Drink()
 
        {
 
            MessageBox.Show("小狗喝水");
 
        }
 
        public override void GotoBed()
 
        {
 
            MessageBox.Show("小狗睡觉");
 
        }
 
        public override 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方法的使用代码示例:
 

    class A
 
    {
 
        public virtual void A1()
 
        {
 
            MessageBox.Show("A---A1");   
 
        }
 
    }
 
    class AA:A
 
    {
 
        public sealed 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方法子类可以对其override或new
 
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类 是一个抽象类,不能被实例化


 

你可能感兴趣的:(C#)