Introduction
Method overriding in C# is a feature like the virtual function in C++. Method overriding is a feature that allows you to invoke functions (that have the same signatures) that belong to different classes in the same hierarchy of inheritance using the base class reference. C# makes use of two keywords: virtual and overrides to accomplish Method overriding. Let's understand this through small examples.
P1.cs
class BC
{
public void Display()
{
System.Console.WriteLine("BC::Display");
}
}
class DC : BC
{
new public void Display()
{
System.Console.WriteLine("DC::Display");
}
}
class Demo
{
public static void Main()
{
BC b;
b = new BC();
b.Display();
}
}
Output
BC::Display
The above program compiles and runs successfully to give the desired output. It consists of a base class BC
and a derived class DC
. Class BC
consists of function Display()
. Class DC
hides the function Display()
it inherited from the base class BC
by providing its on implementatin of Display()
. Class Demo
consists of entrypoint function Main()
. Inside Main()
we first create a reference b
of type BC
. Then we create an object of type BC
and assign its reference to reference variable b
. Using the reference variable b
we invoke the function Display()
. As expected, Display()
of class BC
is executed because the reference variable b
refers to the object of class BC
.
Now we add a twist of lime to the above program.
P2.cs
class BC
{
public void Display()
{
System.Console.WriteLine("BC::Display");
}
}
class DC : BC
{
new public void Display()
{
System.Console.WriteLine("DC::Display");
}
}
class Demo
{
public static void Main()
{
BC b;
b = new BC();
b.Display();
b = new DC();
b.Display();
}
}
Output
BC::Display
BC::Display
Here we are creating an object of Derived class DC
and storing its reference in the reference variable b
of type BC
. This is valid in C#. Next, using the reference variable b
we invoke the function Display()
. Since b
contains a reference to object of type DC
one would expect the function Display()
of class DC
to get executed. But that does not happen. Instead what is executed is the Display()
of BC
class. That's because the function is invoked based on type of the reference and not to what the reference variable b
refers to. Since b
is a reference of type BC
, the function Display()
of class BC
will be invoked, no matter whom b
refers to. Take one more example.
P3.cs
class BC
{
public void Display()
{
System.Console.WriteLine("BC::Display");
}
}
class DC : BC
{
new public void Display()
{
System.Console.WriteLine("DC::Display");
}
}
class TC : BC
{
new public void Display()
{
System.Console.WriteLine("DC::Display");
}
}
class Demo
{
public static void Main()
{
BC b;
b = new BC();
b.Display();
b = new DC();
b.Display();
b = new TC();
b.Display();
}
}
Output
BC::Display
BC::Display
BC::Display
The output of the above program is a receipt of the fact that no matter to whom base class reference b
refers, it invokes the functions of the class that matches its type. But doesn't this sound absurd? If b
contains the reference to a perticular derived class object, then its supposed to invoke the function of that class. Well, C# helps us do this by the usage of keywords virtual and override as shown in the following program.
P4.cs
class BC
{
public virtual void Display()
{
System.Console.WriteLine("BC::Display");
}
}
class DC : BC
{
public override void Display()
{
System.Console.WriteLine("DC::Display");
}
}
class Demo
{
public static void Main()
{
BC b;
b = new BC();
b.Display();
b = new DC();
b.Display();
}
}
Output
BC::Display
DC::Display
The above program compiles and runs successfully to give the expected desired output. The function Display()
of Base class BC
is declared as virtual, while the Derived class's implementation of Display()
is decorated with the modifier override. Doing so enables C# to invoke functions like Display()
based on objects the reference variable refers to and not the type of reference that is invoking the function. Hence in the above program when b
refers to the object of class BC
it invokes Display()
of BC
and then when b
refers to the object of class DC
it invokes Display()
of class DC
. Let's see if this holds true for the third generation of derived classes. Take the following program.
P4.cs
class BC
{
public virtual void Display()
{
System.Console.WriteLine("BC::Display");
}
}
class DC : BC
{
public override void Display()
{
System.Console.WriteLine("DC::Display");
}
}
class TC : DC
{
public override void Display()
{
System.Console.WriteLine("TC::Display");
}
}
class Demo
{
public static void Main()
{
BC b;
b = new BC();
b.Display();
b = new DC();
b.Display();
b = new TC();
b.Display();
}
}
Output
BC::Display
DC::Display
TC::Display
The above program compiles and runs successfully to give the expected desired output. The function Display()
of Base class BC
is declared as virtual, while the implementation of Display()
in successive Derived classes is decorated with the modifier override. Next, we succesively create objects of each class and store their reference in base class reference variable b
and invoke Display()
. The rite versions of Display
get invoked based on the object the reference variable refers to. Time for a tiny teaser! Guess what the output would be in the following program?
P5.cs
class BC
{
public virtual void Display()
{
System.Console.WriteLine("BC::Display");
}
}
class DC : BC
{
public override void Display()
{
System.Console.WriteLine("DC::Display");
}
}
class TC : DC
{
}
class Demo
{
public static void Main()
{
BC b;
b = new TC();
b.Display();
}
}
Output
DC::Display
Since TC
has no implementation of Display()
, it inherits Display()
from DC
as TC
is derived from DC
. Hence Display()
from Derived class DC
gets executed. It's as if the derived class TC
looked like this:
class TC
{
public override void Display()
{
System.Console.WriteLine("DC::Display");
}
}
to the compiler. Take one more example. Guess what its output will be.
P6.cs
class BC
{
public virtual void Display()
{
System.Console.WriteLine("BC::Display");
}
}
class DC : BC
{
public override void Display()
{
System.Console.WriteLine("DC::Display");
}
}
class TC : DC
{
public new void Display()
{
System.Console.WriteLine("TC::Display");
}
}
class Demo
{
public static void Main()
{
BC b;
b = new TC();
b.Display();
}
}
Output
DC::Display
Agreed that TC
defines its own new version of Display()
. But its version of display is not invoked as Display()
of TC
does not override the Display()
of the base class. With this understood we are done with Method overriding in C#.