c#中的interface abstract virtual override和new

from:(http://hi.baidu.com/21tian/blog/item/aabfae6ed4a675de81cb4ab4.html)


interface用来声明接口

1.只提供方法的规则约束,不提供方法的主体。

例:

public interface IPerson

{

void getName();//不包含主体方法

}

2.方法不能用public abstract等修饰,无字段变量,无构造函数。

3.方法可包含参数

public interface IPerson

{

void getAge(string s);

}

例1

public interface IPerson

{

IPerson();//错误

string name;//错误

public void getIDcard();//错误

void getName();//正确

void getAge(string s);//正确

}

实现interface的类

1.与继承类的格式一致,如 public class Chinese:IPerson{}

2.必须实现interface中的各个方法

例2,继承例1

public class Chinese:IPerson

{

public Chinese(){}//添加构造函数

public void getName(){}//实现getName()

public void getAge(){}//实现getAge()

}

abstract用来声明抽象类、抽象方法

1.抽象方法所在类必须为抽象类。

2.抽象类不能直接实例化,必须由其派生类实现。

3.抽象方法不包含方法主体,必须由派生类以override方式实现此方法,这点跟interface中的方法类似


public abstract class Book
{
public Book()
{   
     }

public abstract void getPrice();      //抽象方法,不含主体
        public virtual void getName()   //虚方法,可覆盖
        {
      Console.WriteLine("this is a test:virtual getName()");
             }
        public virtual void getContent()   //虚方法,可覆盖
        {
      Console.WriteLine("this is a test:virtual getContent()");
             }
public void getDate()                           //一般方法,若在派生类中重写,须使用new关键字
        {
      Console.WriteLine("this is a test: void getDate()");
             }
}

public class JavaBook:Book
{
      public override void getPrice() //实现抽象方法,必须实现
      {
Console.WriteLine("this is a test:JavaBook override abstract getPrice()");
}
      public override void getName()   //覆盖原方法,不是必须的
      {
Console.WriteLine("this is a test:JavaBook override virtual getName()");
}
}

测试如下:
public class test
   {
    public test()
    {
     JavaBook jbook=new JavaBook();
     jbook.getPrice();      //将调用JavaBook中getPrice()
     jbook.getName();      //将调用JavaBook中getName()
    jbook.getContent();    //将调用Book中getContent()
     jbook.getDate();       //将调用Book中getDate()
    
    }
    public static void Main()
    {
test t=new test();
    }
   }

virtual标记方法为虚方法
1.可在派生类中以override覆盖此方法
2.不覆盖也可由对象调用
3.无此标记的方法(也无其他标记),重写时需用new隐藏原方法

abstract与virtual: 方法重写时都使用 override 关键字
interface中的方法和abstract方法都要求实现

overrid和new

//override不能重写非虚方法或静态方法。
//所重写的基方法必须是虚拟的、抽象的或重写的。
//所以说虚(virtual)成员与重写(override)成员是配对使用的,
//虽然也可以用new来"隐藏"虚成员

//下面通过例子来看看重写与隐藏的区别,从而进一步理解虚成员的作用。
//重写部分----------

public class MyBaseClass
{
  public virtual void DoSomething()
     {
       Console.WriteLine("Base imp");
     }
}

public class MyDerivedClass:MyBaseClass
{
  public override void DoSomething()
     (
       Console.WriteLine("Derived imp");
     }
}

//其中重写方法将替换基类中的执行代码,这样下面的代码就将使用替换代码,
//即使这是通过基类进行的,情况也是这样:

     MyDerivedClass myObj=new MyDerivedClass();
     MyBaseClass myBaseObj;
     myBaseObj=myObj;
     myBaseObj.DoSomething();
  
//结果如下:

     Derived imp
  

//另外,还可以使用下面的代码隐藏基类方法:
public class MyBaseClass
{
public virtual void DoSomething()
     {
       Console.WriteLine("Base imp");
     }
}

public class MyDerivedClass:MyBaseClass
{
  new public void DoSomething()
     {
        Console.WriteLine("Derived imp");
     }
}
//基类方法不必是虚拟的,但结果是一样的,上面的代码只需要修改行。
//对于基类的虚拟方法和非虚拟方法来说,其结果如下:
       Base imp
//尽管隐藏了基类的执行代码,但仍可以通过基类访问它。

(4) 使用 new 修饰符隐藏基类成员


使用 new 修饰符可以显式隐藏从基类继承的成员。若要隐藏继承的成员,请使用相同名称在派生类中声明该成员,并用 new 修饰符修饰它。


请看下面的类:

public class MyBase
{
public int x ;
public void MyVoke() ;
}


在派生类中用 MyVoke名称声明成员会隐藏基类中的 MyVoke方法,即:

public class MyDerived : MyBase
{ new public void MyVoke (); }


但是,因为字段 x 不是通过类似名隐藏的,所以不会影响该字段。


通过继承隐藏名称采用下列形式之一:


a、引入类或结构中的常数、指定、属性或类型隐藏具有相同名称的所有基类成员。


b、引入类或结构中的方法隐藏基类中具有相同名称的属性、字段和类型。同时也隐藏具有相同签名的所有基类方法。


c、引入类或结构中的索引器将隐藏具有相同名称的所有基类索引器。


注意:在同一成员上同时使用 new 和 override 是错误的。同时使用 new 和 virtual 可保证一个新的专用化点。在不隐藏继承成员的声明中使用 new 修饰符将发出警告。


示例1:在该例中,基类 MyBaseC 和派生类 MyDerivedC 使用相同的字段名 x,从而隐藏了继承字段的值。该例说明了 new 修饰符的使用。同时也说明了如何使用完全限定名访问基类的隐藏成员。

using System ;
public class MyBase
{
public static int x = 55 ;
public static int y = 22 ;
}
public class MyDerived : MyBase
{
new public static int x = 100; // 利用new 隐藏基类的x
public static void Main()
{
// 打印x:
Console.WriteLine(x);
//访问隐藏基类的 x:
Console.WriteLine(MyBase.x);
//打印不隐藏的y:
Console.WriteLine(y);
}
}


输出: 100 55 22


如果移除 new 修饰符,程序将继续编译和运行,但您会收到以下警告:


The keyword new is required on ''MyDerivedC.x'' because it hides inherited member ''MyBaseC.x''.


如果嵌套类型正在隐藏另一种类型,如下例所示,也可以使用 new 修饰符修改此嵌套类型。

原文地址:https://www.cnblogs.com/baishahe/p/1032084.html