Skip to content
Home C# / .NET C# Inheritance — Missing Override Causes Silent Failure

C# Inheritance — Missing Override Causes Silent Failure

Where developers are forged. · Structured learning · Free forever.
📍 Part of: OOP in C# → Topic 2 of 10
Missing override on a new method silently applied base premium discount logic instead of derived.
⚙️ Intermediate — basic C# / .NET knowledge assumed
In this tutorial, you'll learn
Missing `override` on a `new` method silently applied base premium discount logic instead of derived.
  • Inheritance models an 'is-a' relationship; if that doesn't hold, use composition.
  • Use override for polymorphic behaviour, never new unless you know what you're doing.
  • Always call base constructor explicitly using : base(...).
✦ Plain-English analogy ✦ Real code with output ✦ Interview questions
Quick Answer
  • Inheritance in C# models an "is-a" relationship between classes
  • Derived class inherits all non-private members from the base class
  • Use override to replace base behaviour; use new to hide it
  • Base constructor must be explicitly called unless it's parameterless
  • Deep hierarchies (5+ levels) are a smell — prefer composition
  • Mistaking new for override silently breaks polymorphic dispatch
🚨 START HERE

Inheritance Quick Debug Cheat Sheet

Five common inheritance problems and their immediate fix
🟡

Derived method not called via interface or base reference

Immediate ActionCheck for `new` vs `override`
Commands
Search code: `grep -rn "new.*MethodName" ./src`
Add `override` and make base method `virtual`
Fix NowMark base method as `virtual` and derived as `override`. Recompile.
🟡

Base constructor throws NullReferenceException

Immediate ActionInspect derived constructor initializer
Commands
Examine `: base(...)` parameters
Add null checks before passing to base
Fix NowGuard parameters: `: base(param ?? throw new ArgumentNullException(...))`
🟡

Abstract class instantiation at runtime

Immediate ActionYou can't — find the concrete derived class
Commands
Use `is` pattern to check actual type: `if (obj is ConcreteType)`
Refactor to use factory pattern
Fix NowReplace `new AbstractClass()` with factory method returning derived instance
🟡

Sealed class used as base unexpectedly

Immediate ActionRemove `sealed` from class declaration or redesign
Commands
Check class definition for `sealed` keyword
If you can't modify, use composition (wrap the sealed class)
Fix NowRemove `sealed` if appropriate; else create a wrapper class that contains the sealed class instance
🟡

Method signature mismatch in derived class

Immediate ActionUse `override` with exact same signature
Commands
Check base method signature (parameter types, return type)
Use IDE to generate override: type `override` and space
Fix NowDelete mismatched method and regenerate correct signature
Production Incident

Missing `override` Keyword Causes Silent Logic Failure

A payment processing system silently used base class logic instead of the derived override because the `new` keyword was used accidentally.
SymptomPremium discount was never applied, but no compile-time or runtime error occurred. The discount method ran the base version instead of the derived override.
AssumptionThe team assumed that any method with the same name and signature in a derived class would automatically be called when the object is used polymorphically (e.g., through a base class reference).
Root causeThe derived class declared the method with new instead of override. The base method was not marked virtual, so override wasn't possible. When called via a base-class reference (Base ref = new Derived()), the base method was invoked because new does not participate in polymorphism.
Fix1. Mark the base method as virtual. 2. Use override in the derived class. 3. Alternatively, ensure all callers use the derived type directly (defeats polymorphism).
Key Lesson
Always mark methods that are meant to be polymorphically overridden as virtual.In the derived class, always use override — never new — unless you explicitly want to hide the base implementation (rare).Enable compiler warnings about 'new' keyword usage (CS0109) and treat them as errors in CI.
Production Debug Guide

Diagnose inheritance-related bugs fast in production

Polymorphic call doesn't execute expected derived methodCheck if base method is virtual. Check derived method uses override, not new. Use is and as to confirm runtime type.
Base class null reference in constructorVerify that derived constructor calls base constructor with valid arguments. Look for : base(...) syntax and ensure parameters are not null.
Cannot convert derived type to base type in generic contextCheck if covariance/contravariance is needed. Ensure the generic parameter is marked out for covariance.
Accessing base member yields unexpected valueIf derived class shadows a field with new, both base.field and derived field exist separately. Use base.field to explicitly access base value.

Every non-trivial C# application has classes that share behaviour. A Customer and an Employee both have a name, an email, and an ID. A Circle and a Rectangle both have an area and a perimeter. If you write those shared properties and methods twice, you've just signed up for double the maintenance, double the bugs, and double the pain every time requirements change. Inheritance is how C# lets you centralise what's shared and specialise what's different — and it's one of the four pillars of object-oriented programming for a very good reason.

The real problem inheritance solves isn't code length — it's conceptual integrity. When a BankAccount and a SavingsAccount share a base, you're modelling the real world accurately. A savings account IS a bank account. That 'is-a' relationship is the signal that inheritance belongs here. Without it, you end up with bloated utility classes, copy-pasted logic, and systems where fixing one bug means hunting down five nearly-identical methods spread across the codebase.

By the end of this article you'll understand how to build a clean inheritance hierarchy in C#, know the difference between virtual, override, and new (and why confusing them causes silent bugs that are a nightmare to trace), and be able to explain to an interviewer exactly when inheritance is the right tool — and when it absolutely isn't.

What is Inheritance in C#?

Inheritance is the mechanism by which a class (the derived class) can reuse and extend the members of another class (the base class). C# supports single class inheritance — a derived class can have only one direct base class. However, interfaces can be inherited multiple times. The syntax uses a colon: class Derived : Base. All non-private members (fields, properties, methods, events) are inherited. Constructors and finalizers are not inherited, but the derived class must call a base constructor. The protected access modifier is commonly used to share members with derived classes while hiding them from external code.

The real power of inheritance comes from polymorphism: you can treat a derived object as an instance of its base type. Code that operates on a base class can work with any derived class without modification. This is a fundamental concept in object-oriented design.

InheritanceDemo.cs · CSHARP
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748
using System;

namespace TheCodeForge.InheritanceDemo
{
    public class Vehicle
    {
        public string Make { get; set; }
        public string Model { get; set; }
        public int Year { get; set; }

        public Vehicle(string make, string model, int year)
        {
            Make = make;
            Model = model;
            Year = year;
        }

        public virtual void Start()
        {
            Console.WriteLine($"{Make} {Model} started.");
        }
    }

    public class Car : Vehicle
    {
        public int Doors { get; set; }

        public Car(string make, string model, int year, int doors)
            : base(make, model, year)
        {
            Doors = doors;
        }

        public override void Start()
        {
            Console.WriteLine($"{Make} {Model} (car with {Doors} doors) started.");
        }
    }

    public class Program
    {
        public static void Main()
        {
            Vehicle myVehicle = new Car("Toyota", "Camry", 2020, 4);
            myVehicle.Start();  // Output: Toyota Camry (car with 4 doors) started.
        }
    }
}
▶ Output
Toyota Camry (car with 4 doors) started.
Mental Model
Mental Model: Inheritance as a "Is-A" Template
A derived class IS A more specific version of its base class.
  • If you can't say 'Dolphin IS A Mammal' truthfully, don't inherit.
  • The base class captures common behavior and data; the derived class specializes.
  • Polymorphism works because every derived object shares the base interface.
  • If the relationship is 'has-a' instead of 'is-a', use composition over inheritance.
📊 Production Insight
In large codebases, go-to-definition on a method might jump to the base class instead of the derived override.
Use 'Call Hierarchy' or 'Find All References' to see the actual implementation invoked at runtime.
If you're debugging a polymorphic call, check the runtime type with GetType() — don't trust the compile-time type alone.
🎯 Key Takeaway
Inheritance is about reuse of contract, not just implementation.
If you reach for inheritance only to reuse code, reconsider — composition is often safer.
The test: ask yourself 'Is Derived truly a subtype of Base?'

Base and Derived Classes: Syntax and Essentials

To create a derived class, place a colon and the base class name after the derived class name. The derived class automatically inherits all non-private members. However, constructors are not inherited: you must explicitly call a base constructor using : base(...). If the base class has a parameterless constructor, it's called implicitly — but it's still good practice to be explicit.

Access modifiers control visibility to derived classes. Use protected for members that should be accessible to derived types but not to external code. Use private protected (C# 7.2+) to allow access only to derived classes that are in the same assembly.

The base keyword inside a derived class allows you to access base class members even if they are overridden or hidden. It's especially useful when you want to extend a base method instead of replacing it completely.

BaseKeywordDemo.cs · CSHARP
123456789101112131415161718192021222324252627282930313233343536373839
using System;

namespace TheCodeForge.InheritanceDemo
{
    public class Animal
    {
        public string Name { get; set; }

        public Animal(string name)
        {
            Name = name;
        }

        public virtual void Speak()
        {
            Console.WriteLine("Animal makes a sound.");
        }
    }

    public class Dog : Animal
    {
        public Dog(string name) : base(name) { }

        public override void Speak()
        {
            base.Speak();  // Optional: call base implementation
            Console.WriteLine($"{Name} barks!");
        }
    }

    class Program
    {
        static void Main()
        {
            Dog dog = new Dog("Rex");
            dog.Speak();
        }
    }
}
▶ Output
Animal makes a sound.
Rex barks!
📊 Production Insight
When a base constructor throws, the derived constructor never runs. This can leave members uninitialized.
A common pattern is to validate parameters in both constructors, but the base constructor runs first.
If the base constructor is expensive (e.g., DB calls), consider lazy initialization or factory methods.
🎯 Key Takeaway
Always call a base constructor explicitly unless it's parameterless.
Use base.MethodName() to extend base behaviour cleanly.
Remember: the base class constructor runs before the derived class body.

Virtual, Override, and New: The Critical Difference

The virtual keyword in the base class declares a method that can be overridden by a derived class. The override keyword in the derived class replaces the base implementation for all callers, including those using the base type reference. This is the heart of polymorphism.

The new keyword, when applied to a method with the same signature, hides the base method instead of overriding it. This means that the method called depends on the compile-time type of the variable, not the runtime type. This is a common source of subtle bugs.

A method that is not marked virtual cannot be overridden. If you attempt to use override on a non-virtual method, you get a compile-time error. You can still use new to shadow it, but that's usually a design smell.

When you override a method, you can call the base implementation using base.MethodName(). This is useful when you want to add behaviour on top of the base class logic.

⚠ Warning: Don't Confuse `new` with `override`
Using new when you intended polymorphism results in the wrong method being called when the object is treated as its base type. This bug is silent — no compile-time warning by default. Enable IDE warning CS0109 and treat it as an error.
📊 Production Insight
Teams frequently debug 'method not called' for hours only to find new instead of override.
The fix: enable .editorconfig rule dotnet_diagnostic.CS0109.severity = error.
In a large service with many derived types, this one mistake can cause data corruption due to incorrect logic path.
🎯 Key Takeaway
Override changes behaviour for everyone; new only for your own type.
When in doubt, use override.
Enable the compiler warning for 'new' — it's a red flag.
When to Use Virtual/Override vs New
IfDerived class should always replace base behaviour, even via base reference
UseUse virtual in base, override in derived
IfDerived class only wants to shadow base method for direct calls
UseUse new — but only if you understand the break in polymorphism
IfBase method is not virtual and you cannot change it
UseMust use new — but reconsider design; consider composition

Constructors and Inheritance: Base Constructor Calls

Constructors are not inherited, but every derived class must ensure that the base class constructor is called. If the base class has a parameterless constructor, it's called implicitly. If the base class only has parameterized constructors, the derived class must explicitly call one using the colon syntax: public Derived(params) : base(params).

The base constructor runs before the derived constructor body. This means if you have initialization logic in both, the base runs first. Understanding this order is critical when properties are set in the base constructor and then modified in the derived constructor.

If the base constructor throws an exception, the derived constructor will not execute. This is a common cause of NullReferenceException when the derived class's field initializers run after the base constructor call (C# initializes fields before the base constructor call, but that's a subtle point).

🔥Did You Know?
Field initializers in the derived class run before the base constructor. That's why you can't use derived instance fields in base constructor calls — they haven't been assigned yet.
📊 Production Insight
If a base constructor does heavy work (e.g., database lookup) and the derived class doesn't need it, you waste resources.
Consider having a protected parameterless constructor in the base for derived classes that can skip heavy initialization.
Another pattern: use factory methods that allow the derived class to control initialization order.
🎯 Key Takeaway
Always explicitly call a base constructor — never rely on implicit parameterless calls if your base has no default.
Field initializers run before base constructor — watch the order.
If base constructor fails, derived is never fully initialized.

Abstract Classes and Sealed Classes

An abstract class cannot be instantiated directly; it serves as a base class that defines a contract and optionally provides common implementation. Abstract methods have no body and must be overridden in any non-abstract derived class. Abstract classes can also have virtual methods, fields, constructors, etc.

A sealed class prevents further inheritance. It cannot be used as a base class. Sealing a class can help with security, performance (the JIT can devirtualize sealed class method calls), and design clarity.

Abstract classes are often used when you want to force derived classes to implement certain methods while providing shared base logic. They are similar to interfaces but can contain state (fields, properties with backing fields) and method bodies.

C# also allows you to seal individual methods in a derived class, preventing further overriding down the hierarchy.

AbstractSealedDemo.cs · CSHARP
1234567891011121314151617181920212223242526272829303132333435363738
using System;

namespace TheCodeForge.InheritanceDemo
{
    public abstract class Shape
    {
        public abstract double Area();

        public void Display()
        {
            Console.WriteLine($"Area: {Area():F2}");
        }
    }

    public sealed class Circle : Shape
    {
        public double Radius { get; }

        public Circle(double radius)
        {
            Radius = radius;
        }

        public override double Area() => Math.PI * Radius * Radius;
    }

    // This would cause compile error: 'cannot derive from sealed type'
    // public class ColoredCircle : Circle { }

    class Program
    {
        static void Main()
        {
            Shape shape = new Circle(5);
            shape.Display(); // Output: Area: 78.54
        }
    }
}
▶ Output
Area: 78.54
💡Pro Tip: Prefer Interfaces Over Abstract Classes for Contracts
Use abstract classes when you want to share implementation. Use interfaces (with default implementations if needed) when you only want to define a contract. C# allows multiple interface inheritance but only single class inheritance.
📊 Production Insight
Sealing a class can give the JIT more optimization freedom. If a class is internal and never instantiated polymorphically, seal it.
Abstract classes with many abstract methods become hard to maintain because every new method forces changes in all derived classes.
Consider the Template Method pattern via abstract classes to enforce a skeleton algorithm in derived classes.
🎯 Key Takeaway
Abstract classes define a base with shared code and required overrides.
Sealed classes prevent inheritance for security and performance.
Think twice before adding a new abstract method — you'll break all existing derived classes.
🗂 Inheritance Keywords Comparison
When to use each keyword for method inheritance
KeywordEffectPolymorphism?Typical Use Case
virtualAllows derived class to overrideYesBase methods expected to be customized
overrideReplaces base virtual methodYesProviding specific behavior
newHides base method (shadowing)NoWhen you must keep same name but don't want polymorphism (rare)
abstractNo implementation; forces derived overrideYes (when overridden)Contract requiring derived to provide behavior
sealedPrevents further override or inheritanceN/ASecurity, performance, design clarity

🎯 Key Takeaways

  • Inheritance models an 'is-a' relationship; if that doesn't hold, use composition.
  • Use override for polymorphic behaviour, never new unless you know what you're doing.
  • Always call base constructor explicitly using : base(...).
  • Seal classes and methods unless you explicitly expect inheritance.
  • Deep inheritance hierarchies are a design smell; prefer flat structures.

⚠ Common Mistakes to Avoid

    Using `new` instead of `override` for polymorphism
    Symptom

    Derived method not called when using base type variable; no compile-time error.

    Fix

    Mark base method as virtual and derived method as override. Remove new. Enable warning CS0109 as error.

    Forgetting to call base class constructor
    Symptom

    Compile error if base has no parameterless constructor; or base state not initialized.

    Fix

    Add : base(...) to derived constructor with required parameters. If base has only a parameterless constructor, implicit call is fine.

    Creating deep inheritance chains (5+ levels)
    Symptom

    Code is hard to understand, fragile to changes, and violates 'composition over inheritance'.

    Fix

    Refactor deep hierarchies into flattened structures using composition and interfaces. Favor has-a over is-a.

    Using inheritance for code reuse alone without logical 'is-a' relationship
    Symptom

    Derived class does not truly represent a subtype — e.g., inheriting from Stack for a Customer list.

    Fix

    Replace with composition: contain the reusable class as a private field and delegate calls. Use interfaces for shared contract.

    Making all methods virtual 'just in case'
    Symptom

    Unnecessary overhead, hard to reason about behaviour, potential security risks.

    Fix

    Only make methods virtual when you explicitly expect and want overriding. Prefer sealed by default (FxCop rule CA1051).

Interview Questions on This Topic

  • QExplain the difference between override and new in C#.Mid-levelReveal
    override is used when a derived class wants to replace a base class virtual method such that the derived implementation is called even when the object is referenced via a base-typed variable. It participates in polymorphism. new hides the base method; the method called depends on the compile-time type. If you have a Base variable pointing to a Derived object, calling a method declared with new in Derived will invoke the Base method, not the Derived one. Use override for polymorphic behavior, new only when you explicitly want to shadow the base method (rarely).
  • QHow does constructor chaining work in inheritance? Can a derived class prevent a base constructor from running?Mid-levelReveal
    Constructor chaining always runs a base constructor before the derived constructor body. The base constructor is either called explicitly with : base(...) or implicitly if there is a parameterless constructor. You cannot prevent the base constructor from running entirely, but you can control which constructor is called by providing different parameters. If the base class has a protected parameterless constructor, the derived class can call it, potentially skipping heavy initialization logic.
  • QWhat is the difference between an abstract class and an interface in C#?SeniorReveal
    Abstract classes can have implementation (methods with bodies), fields, constructors, and access modifiers. They are ideal for sharing common code among closely related classes. Interfaces (pre-C# 8.0) could only declare members without implementation; since C# 8.0, interfaces can have default implementations, but they still cannot hold state (fields). A class can inherit only one abstract class but multiple interfaces. Abstract classes are 'is-a' relationships; interfaces are 'can-do' contracts. Use abstract classes when subclasses share a common base implementation; use interfaces to define capabilities across unrelated types.
  • QCan you override a method that is not marked virtual in the base class?JuniorReveal
    No, you cannot override a non-virtual method. The C# compiler will produce an error if you try to use override. You can, however, use new to hide it, but that does not participate in polymorphism. If you need to override, you must change the base class to mark the method virtual (or abstract). If you cannot modify the base class, you must either use new and accept the limitations, or use composition with a wrapper class.

Frequently Asked Questions

What is inheritance in C# in simple terms?

Inheritance lets a class (derived) automatically get all non-private members from another class (base). You can then add or modify behaviour. It's like a child inheriting traits from a parent, but with the ability to learn new skills.

Can a class inherit from multiple classes in C#?

No, C# supports only single class inheritance. However, a class can implement multiple interfaces. This avoids the diamond problem. If you need multiple base behaviours, use interfaces or composition.

What is the base keyword used for in C#?

base is used in a derived class to access members of the base class. It's commonly used to call base constructors (: base()) or to invoke the base implementation of an overridden method (base.SomeMethod()).

What is the difference between hiding and overriding?

Hiding (with new) creates a new method with the same name; the original base method still exists. Which one gets called depends on the compile-time type of the variable. Overriding (with override) replaces the base method entirely so that the derived version is always called, regardless of the variable type. Hiding is non-polymorphic; overriding is polymorphic.

When should I use abstract classes vs interfaces?

Use an abstract class when derived classes share common state or implementation logic (e.g., a base Animal with a MakeSound method implemented for all animals). Use an interface when you want to define a contract that can be adopted by unrelated classes (e.g., IDisposable). Since C# 8, interfaces can have default implementations, but they cannot have fields or constructors.

🔥
Naren Founder & Author

Developer and founder of TheCodeForge. I built this site because I was tired of tutorials that explain what to type without explaining why it works. Every article here is written to make concepts actually click.

← PreviousClasses and Objects in C#Next →Interfaces in C#
Forged with 🔥 at TheCodeForge.io — Where Developers Are Forged