C# Inheritance — Missing Override Causes Silent Failure
Missing override on a new method silently applied base premium discount logic instead of derived.
20+ years shipping production .NET services in enterprise systems. Lessons pulled from things that broke in production.
- Inheritance in C# models an "is-a" relationship between classes
- Derived class inherits all non-private members from the base class
- Use
overrideto replace base behaviour; usenewto hide it - Base constructor must be explicitly called unless it's parameterless
- Deep hierarchies (5+ levels) are a smell — prefer composition
- Mistaking
newforoverridesilently breaks polymorphic dispatch
Think of a vehicle blueprint. Every vehicle has wheels, an engine, and a way to move. A car is a vehicle, but it also has a specific number of doors and a trunk. A motorcycle is also a vehicle, but it has two wheels and no roof. Inheritance lets you write the 'vehicle' stuff once and let 'Car' and 'Motorcycle' borrow it automatically — then each adds only what makes it unique. You're not copying and pasting; you're saying 'start with everything a Vehicle already knows, and add on from there.'
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.
- 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.
GetType() — don't trust the compile-time type alone.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.
base.MethodName() to extend base behaviour cleanly.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.. This is useful when you want to add behaviour on top of the base class logic.MethodName()
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.new instead of override.dotnet_diagnostic.CS0109.severity = error.virtual in base, override in derivednew — but only if you understand the break in polymorphismnew — but reconsider design; consider compositionConstructors 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).
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.
The “Is-A” Rule: Why You’ll Burn Yourself Without It
Inheritance isn't a code-sharing shortcut. It models an “is-a” relationship. A Dog is an Animal. A Manager is an Employee. If that sentence sounds wrong in plain English, don't use inheritance. I've cleaned up diffs where a junior inherited from List<T> just to get Add() for free. That's not an “is-a”; that's a design debt. The base class defines a contract. If your derived type can't fulfill every public member of the base without lying, you need composition or interfaces instead. Nothing kills a codebase faster than a Rectangle inheriting from Square because some tutorial said “reuse.” Real production: I once saw a SqlConnection wrapper inherit from Stream because the original author wanted Read(). Result: phantom open connections and NotSupportedException everywhere. Ask yourself: “Will this derived class ever violate a base class invariant?” If yes, stop. Use an interface. The compiler won't save you from bad semantics.
Types of Inheritance C# Actually Gives You (and One It Doesn't)
C# supports exactly three inheritance shapes directly. Single: one class, one parent. Multilevel: A -> B -> C — the chain you see in ORM entities or layered services. Hierarchical: one base, multiple children (e.g., PaymentMethod with CreditCard and PayPal). That's it. No multiple inheritance of classes. The compiler enforces this because diamond problems cause runtime heisenbugs. The other patterns you need? Interfaces. They solve the “what” without the “how” coupling. You can implement IDisposable, IComparable, and ISerialize on the same class—that's multiple inheritance of contract. In .NET 8, you can also do default interface methods for shared behavior without the base class baggage. Production example: a NotificationService that implements both IEmailSender and ISmsSender. No shared state, no base class coupling, just clean polymorphism. If you need true multiple inheritance of behavior, stop and refactor to composition or facade pattern.
Missing `override` Keyword Causes Silent Logic Failure
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.virtual.
2. Use override in the derived class.
3. Alternatively, ensure all callers use the derived type directly (defeats polymorphism).- Always mark methods that are meant to be polymorphically overridden as
virtual. - In the derived class, always use
override— nevernew— 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.
virtual. Check derived method uses override, not new. Use is and as to confirm runtime type.: base(...) syntax and ensure parameters are not null.out for covariance.new, both base.field and derived field exist separately. Use base.field to explicitly access base value.Search code: `grep -rn "new.*MethodName" ./src`Add `override` and make base method `virtual`virtual and derived as override. Recompile.Key takeaways
override for polymorphic behaviour, never new unless you know what you're doing.Common mistakes to avoid
5 patternsUsing `new` instead of `override` for polymorphism
virtual and derived method as override. Remove new. Enable warning CS0109 as error.Forgetting to call base class constructor
: base(...) to derived constructor with required parameters. If base has only a parameterless constructor, implicit call is fine.Creating deep inheritance chains (5+ levels)
has-a over is-a.Using inheritance for code reuse alone without logical 'is-a' relationship
Stack for a Customer list.Making all methods virtual 'just in case'
Interview Questions on This Topic
Explain the difference between `override` and `new` in C#.
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).Frequently Asked Questions
20+ years shipping production .NET services in enterprise systems. Lessons pulled from things that broke in production.
That's OOP in C#. Mark it forged?
6 min read · try the examples if you haven't