冠冕堂皇

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::

在C#的学习中,容易混淆virtual方法和abstract方法的使用,现在来讨论一下二者的区别。二者都牵涉到在派生类中与override的配合使用。

一、Virtual方法(虚方法)

     virtual 关键字用于在基类中修饰方法。virtual的使用会有两种情况:

     情况1:在基类中定义了virtual方法,但在派生类中没有重写该虚方法。那么在对派生类实例的调用中,该虚方法使用的是基类定义的方法。

     情况2:在基类中定义了virtual方法,然后在派生类中使用override重写该方法。那么在对派生类实例的调用中,该虚方法使用的是派生重写的方法。

二、Abstract方法(抽象方法)

     abstract关键字只能用在抽象类中修饰方法,并且没有具体的实现。抽象方法的实现必须在派生类中使用override关键字来实现

 
虚方法和抽象方法都可以供派生类重写,它们之间有什么区别呢?
1. 虚方法必须有实现部分(好像这句话有误),抽象方法没有提供实现部分,抽象方法是一种强制派生类覆盖的方法,否则派生类将不能被实例化。如:
//抽象方法
public abstract class Animal
{
    public abstract void Sleep();
    public abstract void Eat();
}

//虚方法
public class Animal
{
    public virtual void Sleep(){}
    public virtual void Eat(){}
}

2. 抽象方法只能在抽象类中声明,虚方法不是。其实如果类包含抽象方法,那么该类也是抽象的,也必须声明为抽象的。如:
public class Animal
{
    public abstract void Sleep();
    public abstract void Eat();
}
 

编译器会报错:
Main.cs(10): 'VSTest.Animal.Sleep()' is abstract but it is contained in nonabstract class 'VSTest.Animal'
Main.cs(11): 'VSTest.Animal.Eat()' is abstract but it is contained in nonabstract class 'VSTest.Animal'

3. 抽象方法必须在派生类中重写,这一点跟接口类似,虚方法不必。如:

public abstract class Animal
{
    public abstract void Sleep();
    public abstract void Eat();
}

public class Cat : Animal
{
    public override void Sleep()
    {
        Console.WriteLine( "Cat is sleeping" );
    }
    // we need implement Animal.Eat() here

}

编译器会报错:Main.cs(14): 'VSTest.Cat' does not implement inherited abstract member 'VSTest.Animal.Eat()',因为我们没有实现抽象类中所有抽象方法。

 

 C#中 abstrac与virtual的区别

1.abstract方法只能在抽象类中声明,虚方法则不是。
   abstract方法必须在派生类中重写,而virtual则不必
2.abstract方法不能声明方法实体,
      abstract public void SD();
   虚方法则可以
      public virtual void sdf()
        {
            Console.WriteLine("A");
        }
3.虚方法可以实现多态,而抽象方法不可以。。。
 
 
c#中的interface abstract 与 virtual
 

 

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();         //right
   void getAge(string s);  //right
}

实现interface的类
1.与继承类的格式一致,如 public class Chinese:IPerson{}
2.必须实现 interface 中的各个方法

   例2,继承例1
public class Chinese:IPerson

   public Chinese(){}                  //添加构造
   public void getName(){}          //实现getName()
   public void getAge(string s){} //实现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方法都要求实现

 

 

 

virtual 关键字用于在基类中修饰方法。virtual的使用会有两种情况:

     情况1:在基类中定义了virtual方法,但在派生类中没有重写该虚方法。那么在对派生类实例的调用中,该虚方法使用的是基类定义的方法。

     情况2:在基类中定义了virtual方法,然后在派生类中使用override重写该方法。那么在对派生类实例的调用中,该虚方法使用的是派生重写的方法。

 二、Abstract方法(抽象方法)

     abstract关键字只能用在抽象类中修饰方法,并且没有具体的实现,只需要一个方法的签名+”;“加”{}“,抽象方法的实现必须在派生类中使用override关键字来实现。

 三、virtual:用于父类中声明该方法可被重写(其父类也可以是抽象类)。

    存在抽象方法的类一定是抽象类,但抽象类中可以无抽象方法。继承抽象类的派生类一定要实现基类的抽象方法。这一点是和虚方法不同的。

  

抽象方法   
   使用abstract关键字    public    abstract    bool    Withdraw(…);   
   抽象方法是必须被派生类覆写的方法。   
   抽象方法是可以看成是没有实现体的虚方法   
   如果类中包含抽象方法,那么类就必须定义为抽象类,不论是否还包含其它一般方法   
    
   虚方法   
   使用virtual关键字    public    virtual    bool    Withdraw(…);   
   调用虚方法,运行时将确定调用对象是什么类的实例,并调用适当的覆写的方法。   
   虚方法可以有实现体

 

若一个实例方法的声明中含有        virtual        修饰符,则称该方法为虚拟方法;一个虚拟方法的实现可以由派生类取代。取代所继承的虚拟方法的实现的过程称为重写该方法;在一个虚拟方法调用中,该调用所涉及的那个实例的运行时类型确定了要被调用的究竟是该方法的哪一个实现。         
            
       虚函数的限制:         
            
       1.虚函数仅适用于有继承关系的类对象,        所以只有类的成员函数才能说明为虚函数.             
       2.静态成员函数不能是虚函数.             
       3.内联函数不能是虚函数.             
       4构造函数不能是虚函数.             
       5.析构函数可以是虚函数.

 

简单点说,抽象方法是需要子类去实现的.虚方法,是已经实现了,子类可以去覆盖,也可以不覆盖取决于需求.   
   如   
   public    abstract    class    AUser   
   {   
           public    abstract    UserInfo    getUser();   
           public    virtual    void    Save(UserInfo    info)   
           {   
                   //实现保存的功能   
           }   
   }   
    
   public    class    UserSqlServer:AUser   
   {   
           public    override    UserInfo    getUser()   
           {   
                     //一定要实现的,抽象类只给了抽象方法   
           }   
           //假设抽象类针对SqlServer实现的功能,这里不需要实现了.   
   }   
    
   public    class    UserOracle:AUser   
   {   
           public    override    UserInfo    getUser()   
           {   
                     //一定要实现的,抽象类只给了抽象方法   
           }   
           //假设抽象类针对Oracle实现的功能   
           public    override    void    Save(UserInfo    info)   
           {   
                     //覆盖抽象类中的Save方法   
           }   
   }   
   以上只

是例子,真实并不这么做.  

 
Abstract和Virtual的区别

Virtual(虚拟):


1.修饰方法、属性、索引器或事件声明


2.允许在派生类中重写这些对象。说明:派生类可以重写这些方法,也可以不重写这些方法


3.修饰方法时注意:(可以有方法体)

l         调用虚方法时,将为重写成员检查该对象的运行时类型。将调用大部分派生类中的该重写成员,如果没有派生类重写该成员,则它可能是原始成员。

l         默认情况下,方法是非虚拟的。不能重写非虚方法。

l         virtual 修饰符不能与 static、abstract, private 或 override 修饰符一起使用。

 

4.虚拟属性的行为与抽象方法一样。

l         在静态属性上使用 virtual 修饰符是错误的。

l         通过包括使用 override 修饰符的属性声明,可在派生类中重写虚拟继承属性。


Abstract(抽象):

1.修饰类、方法、属性、索引器及事件

2.在类声明中使用 abstract 修饰符以指示某个类只能是其他类的基类。

3.标记为抽象或包含在抽象类中的成员必须通过从抽象类派生的类来实现。说明:派生类必须实现

4.抽象类具有以下特性:

·      抽象类不能实例化。

·      抽象类可以包含抽象方法和抽象访问器。

·      不能用 sealed(C# 参考)修饰符修改抽象类,这意味着抽象类不能被继承。

·      从抽象类派生的非抽象类必须包括继承的所有抽象方法和抽象访问器的实实现。

在方法或属性声明中使用 abstract 修饰符以指示方法或属性不包含实现。

5.抽象方法具有以下特性:(没有方法体)

·      抽象方法是隐式的虚方法。

·      只允许在抽象类中使用抽象方法声明。

·      因为抽象方法声明不提供实际的实现,所以没有方法体;方法声明只是以一个分号结束,并且在签名后没有大括号 ({ })。例如: public abstract void MyMethod();

·      实现由一个重写方法override(C# 参考)提供,此重写方法是非抽象类的一个成员。

·      在抽象方法声明中使用 static 或 virtual 修饰符是错误的。

6.抽象属性的行为与抽象方法一样。

·      在静态属性上使用 abstract 修饰符是错误的。

在派生类中,通过包括使用 override 修饰符的属性声明,可以重写抽象的继承属性。

posted on 2013-06-07 17:45  冠冕堂皇  阅读(780)  评论(0编辑  收藏  举报