C#类中方法的执行顺序
有些中级开发小伙伴还是搞不太明白在继承父类以及不同场景实例化的情况下,父类和子类的各种方法的执行顺序到底是什么,下面通过场景的举例来重新认识下方法的执行顺序:
(下面内容涉及到了C#中的继承,构造函数,虚方法,虚方法的重写,new关键字等知识点)
场景一
有子类继承,但是只实例化父类:只执行A对象,输出A对象的信息
class A
{
public A() => Console.WriteLine("A的构造函数");
public virtual void Fun() => Console.WriteLine("A的方法");
}
class B : A
{
public B() => Console.WriteLine("B的构造函数");
public void Fun() => Console.WriteLine("B的方法");
}
class Program
{
static void Main(string[] args)
{
A a = new A();
a.Fun();
Console.ReadLine();
}
}
上述Main方法中在new A对象时,程序首先进入class A中,执行class A的构造函数A(),然后执行class A中的Fun()方法,故运行结果为:
场景二
实例化子类,子类和父类的构造函数的执行顺序:当执行B对象时,因为继承A对象,所以首先执行基类A的构造函数
class A
{
public A() => Console.WriteLine("A的构造函数");
public virtual void Fun()=> Console.WriteLine("A的方法");
}
class B : A
{
public B() => Console.WriteLine("B的构造函数");
public void Fun() => Console.WriteLine("B的方法");
}
class Program
{
static void Main(string[] args)
{
B b = new B();
b.Fun();
Console.ReadKey();
}
}
上述Main方法中在new B对象时,由于B继承A,先执行父类的构造函数,所以先执行A中的构造函数A(),然后在执行B中的构造函数B(),故运行结果为:
场景三
父类有虚方法,子类没有使用(override)关键字重写父类方法的时候,使用的是new关键字时:
class A
{
public A()=> Console.WriteLine("A的构造函数");
public virtual void Fun() => Console.WriteLine("A的方法");
}
class B : A
{
public B() => Console.WriteLine("B的构造函数");
//不写new时,该方法会抛出警告,但不是错误
public new void Fun()=> Console.WriteLine("B的方法");
}
class Program
{
static void Main(string[] args)
{
A a = new B();
a.Fun();
Console.ReadKey();
}
}
上述Main方法中先new B对象,先执行A中的构造函数A(),然后在执行B中的构造函数B(),最后调用class A的Fun()方法(没有重写父类方法),故运行结果为:
场景四
父类有虚方法, 当子类重写了(override)父类的方法时:
class A
{
public A()=> Console.WriteLine("A的构造函数");
public virtual void Fun() => Console.WriteLine("A的方法");
}
class B : A
{
public B()=> Console.WriteLine("B的构造函数");
public override void Fun()=> Console.WriteLine("B的方法");
}
static void Main(string[] args)
{
A a = new B();
a.Fun();
Console.ReadKey();
}
上述Main方法同样是先new B对象,先执行A中的构造函数A(),然后在执行B中的构造函数B(),但是子方法中使用了override关键字“覆盖”,使得子类中方法覆盖了父类中的方法,无法再访问父类中原始方法。(要重写方法,父类方法必须有virtual关键字),所以其运行结果为:
场景五
基类是接口层,多重继承时:
interface I
{
void Fun();
}
class A : I
{
public A() => Console.WriteLine("A的构造函数");
public virtual void Fun() => Console.WriteLine("A的方法");
}
class B : A
{
public B() => Console.WriteLine("B的构造函数");
//不写new时,该方法会抛出警告
public new void Fun() =>Console.WriteLine("B的方法");
}
static void Main(string[] args)
{
B b = new B();
b.Fun();
((A)b).Fun();
((I)b).Fun();
Console.ReadKey();
}
打印结果:
场景六
当多重继承,子类重写override父类方法时:
interface I
{
void Fun();
}
class A : I
{
public A() => Console.WriteLine("A的构造函数");
public virtual void Fun() => Console.WriteLine("A的方法");
}
class B : A
{
public B() => Console.WriteLine("B的构造函数");
public override void Fun() =>Console.WriteLine("B的方法");
}
static void Main(string[] args)
{
B b = new B();
b.Fun();
((A)b).Fun();
((I)b).Fun();
Console.ReadKey();
}
打印结果:(对比场景5)
场景七
使用new重写父类方法,同时让每个子类都继承接口:
interface I
{
void Fun();
}
class A : I
{
public A() => Console.WriteLine("A的构造函数");
public virtual void Fun() => Console.WriteLine("A的方法");
}
class B : A, I
{
public B() => Console.WriteLine("B的构造函数");
//不写new时,该方法会抛出警告
public new void Fun() => Console.WriteLine("B的方法");
}
static void Main(string[] args)
{
B b = new B();
b.Fun();
((A)b).Fun();
((I)b).Fun();
Console.ReadKey();
}
打印结果:
最新文章
- Drools 查询学习
- BootStrap入门教程 (一)
- IOS第四天(2:字典转模型plist)
- hdu-1789-Doing Homework again
- python操作mongodb之四cp数据库
- Hibernate 通过 Session 操纵对象
- [Effective C++ --028]避免返回handles指向对象内部成分
- 对Android中dp单位的理解
- logstash multiline
- Spring整合的quartz任务调度的实现方式
- Codeforce 57C Array
- OpenCV249 for python278 最简配置方案
- 第七章——集成学习和随机森林(Ensemble Learning and Random Forests)
- web优化(二)
- 数据挖掘---Pandas的学习
- SpringMVC拦截器与异常处理
- nginx增加ssl支持 - 编译时参数详情列表
- ResultJsonInfo<;T>;
- 20145203Java实验报告四:Android开发基础
- WebSocket client for python
热门文章
- java例题_44 一个偶数总能表示为两个素数之和
- 配置redis 4.0.11 集群
- [hash]集合
- 北航OO第二单元作业总结(2.1~2.3)
- Java代码度量分析工具:DesigniteJava简介
- Oracle-buffer cache过小导致SQL执行时间长
- SpringBoot+Dubbo+Zookeeper 实例
- ssh+scp基本使用
- Mysql 添加 create_time, update_time 创建时间 更新时间 自动更新
- 为什么 DNS 协议使用 UDP?只使用了 UDP 吗?