C#类中虚方法相互调用的潜在重载错误

简介:
当我们编写基类虚方法时,需要注意一个问题,就是基类中虚方法的相互调用,有可能引起派生类重载时的潜在错误隐患。当然这个错误并不是C#语言设计的缺陷,而是一个不可避免的实现而已。当然如果我们是要编写通用的组建基类,就需要注意一下了。

    或许我们刚开始做OOP的时候,对于有没有方法有没有virtual根本不在乎,很多是时候我们都重写了(rewrite)了基类方法。当然在需要确定重载(override)的时候,virtual关键字限定基类方法是不可少的。那么是不时我们就可以把基类的方法都弄成virtual修饰的呢?这样虽然在大多时候没有问题,而其如果是自己重载自己的基类出问题的可能性也不大,可是如果是别人来继承基类,那么问题可能就来了。

    当我们重载不确切的基类是,最好的习惯是调用以下 base的同名方法,这个在控件开发时用的更加普遍。可是这个时候,如果基类之间存在虚方法调用了别的被重载的虚方法,潜在错误就出来了。看下面示例(由 xingd提供,我修改):
using System;

public  class Base
{
     public  virtual  void Foo()
    {
        Console.WriteLine("Base::Foo");
         this.Bar();
    }
     public  virtual  void Bar()
    {
        Console.WriteLine("Base::Bar");
    }
};

public  class Derived : Base
{
     private Object obj;
     public  override  void Foo()
    {
        Console.WriteLine("Derived::Foo");
         base.Foo();
        obj =  new Object();
         this.Bar();
    }
     public  override  void Bar()
    {
        Console.WriteLine(obj.ToString());
        Console.WriteLine("Derived::Bar");
    }
};

public  class Test
{
     public  static  void Main()
    {
        Derived b =  new Derived();
        b.Foo();
    }
};

    编译没有错误,当然了,又不是在里讲C#语法 。运行结果为:
   E:\Working\Doing>test
   Derived::Foo
   Base::Foo

   Unhandled Exception: System.NullReferenceException: Object reference not set to an instance of an object.
     at Derived.Bar()
     at Derived.Foo()
     at Test.Main()

    问题代码就是:
        Console.WriteLine("Derived::Foo");
         base.Foo();
        obj =  new Object();
         this.Bar();
    
    由于 base.Foo()中的 this.Bar()方法已被重载,所以实际执行的是Derived::Bar,而这个时候我的obj还没有初始化呢。

    当然修正这个bug也很容易,就是把Base::Bar变为非虚方法就可以了。由于C#可以灵活的设置virtual来控制方法是否需要重载,所以这类问题完全是由代码的设计缺陷所引起的,而且如果由一个人来写一般是不会设计出这样的代码的,可是如果基类和派生类由不同的人来写,出这样的错误的机会可能就会更大一些。


本文转自博客园鸟食轩的博客,原文链接:http://www.cnblogs.com/birdshome/,如需转载请自行联系原博主。

目录
相关文章
|
18天前
|
开发框架 .NET C#
C#|.net core 基础 - 删除字符串最后一个字符的七大类N种实现方式
【10月更文挑战第9天】在 C#/.NET Core 中,有多种方法可以删除字符串的最后一个字符,包括使用 `Substring` 方法、`Remove` 方法、`ToCharArray` 与 `Array.Copy`、`StringBuilder`、正则表达式、循环遍历字符数组以及使用 LINQ 的 `SkipLast` 方法。
|
1月前
|
存储 C# 索引
C# 一分钟浅谈:数组与集合类的基本操作
【9月更文挑战第1天】本文详细介绍了C#中数组和集合类的基本操作,包括创建、访问、遍历及常见问题的解决方法。数组适用于固定长度的数据存储,而集合类如`List<T>`则提供了动态扩展的能力。文章通过示例代码展示了如何处理索引越界、数组长度不可变及集合容量不足等问题,并提供了解决方案。掌握这些基础知识可使程序更加高效和清晰。
71 2
|
15天前
|
Java 程序员 C#
【类的应用】C#应用之派生类构造方法给基类构造方法传参赋值
【类的应用】C#应用之派生类构造方法给基类构造方法传参赋值
8 0
|
1月前
|
C# 数据安全/隐私保护
C# 一分钟浅谈:类与对象的概念理解
【9月更文挑战第2天】本文从零开始详细介绍了C#中的类与对象概念。类作为一种自定义数据类型,定义了对象的属性和方法;对象则是类的实例,拥有独立的状态。通过具体代码示例,如定义 `Person` 类及其实例化过程,帮助读者更好地理解和应用这两个核心概念。此外,还总结了常见的问题及解决方法,为编写高质量的面向对象程序奠定基础。
23 2
|
2月前
|
C#
C#中的类和继承
C#中的类和继承
39 6
|
2月前
|
Java C# 索引
C# 面向对象编程(一)——类
C# 面向对象编程(一)——类
32 0
|
2月前
|
开发框架 .NET 编译器
C# 中的记录(record)类型和类(class)类型对比总结
C# 中的记录(record)类型和类(class)类型对比总结
|
4月前
|
开发框架 .NET 编译器
程序与技术分享:C#基础知识梳理系列三:C#类成员:常量、字段、属性
程序与技术分享:C#基础知识梳理系列三:C#类成员:常量、字段、属性
34 2
|
4月前
|
C#
C# 版本的 计时器类 精确到微秒 秒后保留一位小数 支持年月日时分秒带单位的输出
这篇2010年的文章是从别处搬运过来的,主要包含一个C#类`TimeCount`,该类有多个方法用于处理时间相关的计算。例如,`GetMaxYearCount`计算以毫秒为单位的最大年数,`GetCurrentTimeByMiliSec`将当前时间转换为毫秒,还有`SecondsToYYMMDDhhmmss`将秒数转换为年月日时分秒的字符串。此外,类中还包括一些辅助方法,如处理小数点后保留一位数字的`RemainOneFigureAfterDot`。
|
4月前
|
存储 安全 C#
C# 类的深入指南
C# 类的深入指南