C# - Interfaces

Interfaces in C#

Interfaces in C# are a fundamental part of the language's support for abstraction and polymorphism. They allow you to define contracts that classes or structs must implement, enabling flexible and loosely coupled designs. This document provides a thorough exploration of interfaces, including their syntax, purpose, advanced features, and practical use cases.

What is an Interface?

An interface is a reference type in C# that defines a set of method signatures, properties, events, or indexers without any implementation. It specifies what a class or struct should do but not how it does it.

By implementing an interface, a class or struct agrees to provide implementations for all members declared in the interface.

Basic Syntax of an Interface

public interface IShape
{
    double GetArea();
    double GetPerimeter();
}

Here, IShape declares two methods that any implementing class must define.

Implementing an Interface

public class Rectangle : IShape
{
    public double Width { get; set; }
    public double Height { get; set; }

    public double GetArea()
    {
        return Width * Height;
    }

    public double GetPerimeter()
    {
        return 2 * (Width + Height);
    }
}

Why Use Interfaces?

Interfaces provide several important benefits:

  • Abstraction: Define a contract without implementation.
  • Polymorphism: Different classes can be treated uniformly if they implement the same interface.
  • Decoupling: Code depending on interfaces rather than concrete classes is more flexible and testable.
  • Multiple Inheritance: C# doesn’t support multiple inheritance of classes, but a class can implement multiple interfaces.

Interface Members

Interfaces can declare several types of members, all of which are implicitly public and abstract (no implementation, unless using default interface methods - see below):

  • Methods
  • Properties
  • Events
  • Indexers

Example with Properties and Events

public interface INotifier
{
    event EventHandler NotificationReceived;

    string Status { get; set; }

    void Notify(string message);
}

Implementing Multiple Interfaces

A class or struct can implement more than one interface, allowing a flexible composition of behaviors.

public interface IFlyable
{
    void Fly();
}

public interface ISwimmable
{
    void Swim();
}

public class Duck : IFlyable, ISwimmable
{
    public void Fly()
    {
        Console.WriteLine("Flying");
    }

    public void Swim()
    {
        Console.WriteLine("Swimming");
    }
}

Explicit Interface Implementation

When a class implements multiple interfaces that have members with the same signature, or when you want to hide interface methods from the public API, you can implement the interface explicitly.

Example

public interface IFirst
{
    void DoWork();
}

public interface ISecond
{
    void DoWork();
}

public class Worker : IFirst, ISecond
{
    void IFirst.DoWork()
    {
        Console.WriteLine("First DoWork");
    }

    void ISecond.DoWork()
    {
        Console.WriteLine("Second DoWork");
    }
}

Usage:

Worker worker = new Worker();

((IFirst)worker).DoWork();   // Output: First DoWork
((ISecond)worker).DoWork();  // Output: Second DoWork

Default Interface Methods (C# 8.0+)

Starting from C# 8.0, interfaces can provide default implementations for members. This allows interface authors to add new members without breaking existing implementations.

public interface ILogger
{
    void Log(string message)
    {
        Console.WriteLine($"Log: {message}");
    }
}

Classes implementing ILogger can override Log but aren’t required to.

Example

public class ConsoleLogger : ILogger
{
    // Uses default Log implementation
}

public class FileLogger : ILogger
{
    public void Log(string message)
    {
        // Custom file logging logic
    }
}

Interface Inheritance

Interfaces can inherit from one or more other interfaces, enabling hierarchical contracts.

public interface IMovable
{
    void Move();
}

public interface IAnimal : IMovable
{
    void Eat();
}

public class Dog : IAnimal
{
    public void Move()
    {
        Console.WriteLine("Dog moves");
    }

    public void Eat()
    {
        Console.WriteLine("Dog eats");
    }
}

Interfaces vs Abstract Classes

Aspect Interface Abstract Class
Purpose Defines a contract with no implementation (except default methods) Defines a partial implementation
Multiple Inheritance Supported (a class can implement multiple interfaces) Not supported (only single inheritance)
Members Methods, properties, events, indexers Can contain fields, constructors, full method implementations
Access Modifiers Members are implicitly public Members can have access modifiers
Versioning Default implementations from C# 8.0 allow better versioning Can add new methods without breaking derived classes

Interface Variables and Polymorphism

Variables of an interface type can reference any object that implements the interface, enabling polymorphism.

public interface IWorker
{
    void Work();
}

public class Developer : IWorker
{
    public void Work()
    {
        Console.WriteLine("Writing code");
    }
}

public class Manager : IWorker
{
    public void Work()
    {
        Console.WriteLine("Managing team");
    }
}

IWorker worker = new Developer();
worker.Work(); // Output: Writing code

worker = new Manager();
worker.Work(); // Output: Managing team

Interface Segregation Principle (ISP)

One of the SOLID design principles states that no client should be forced to depend on methods it does not use. Interfaces enable this by breaking large interfaces into smaller, more specific ones.

Example

public interface IPrinter
{
    void Print();
}

public interface IScanner
{
    void Scan();
}

public class MultiFunctionPrinter : IPrinter, IScanner
{
    public void Print() { /* ... */ }
    public void Scan() { /* ... */ }
}

public class SimplePrinter : IPrinter
{
    public void Print() { /* ... */ }
}

Interfaces with Generics

Interfaces can be generic, allowing type parameters to specify flexible contracts.

public interface IRepository
{
    void Add(T item);
    T Get(int id);
}
public class UserRepository : IRepository
{
    public void Add(User item) { /* ... */ }
    public User Get(int id) { /* ... */ return new User(); }
}

Interfaces and Dependency Injection (DI)

Interfaces are widely used in DI frameworks to inject dependencies, enhancing testability and modularity.

// Define an interface
public interface IMessageService
{
    void SendMessage(string message);
}

// Implement the interface
public class EmailService : IMessageService
{
    public void SendMessage(string message)
    {
        Console.WriteLine("Sending email: " + message);
    }
}

// Consumer class
public class Notification
{
    private readonly IMessageService _messageService;

    public Notification(IMessageService messageService)
    {
        _messageService = messageService;
    }

    public void Notify(string message)
    {
        _messageService.SendMessage(message);
    }
}

Explicit Interface Implementation and Access Control

Explicit interface implementation allows methods to be accessible only via interface references, hiding them from the public API of the class.

public interface ILogger
{
    void Log(string message);
}

public class SilentLogger : ILogger
{
    void ILogger.Log(string message)
    {
        // Does nothing (silently ignores logs)
    }
}

SilentLogger logger = new SilentLogger();
// logger.Log("Hi"); // Error: not accessible
((ILogger)logger).Log("Hi"); // Works fine

Interfaces with Default Implementations (Example)

public interface IShape
{
    double GetArea();

    double GetPerimeter()
    {
        return 0; // Default implementation returns zero
    }
}

Limitations of Interfaces

  • Interfaces cannot contain fields (state).
  • Prior to C# 8, interfaces could not contain any implementation.
  • Interfaces can become difficult to maintain if overused or poorly designed.
  • Versioning can be challenging without default implementations.

Interface Evolution in C#

Interfaces have evolved significantly:

  • Early C#: Interfaces only declared signatures.
  • C# 8: Added default interface methods, static methods in interfaces.
  • C# 11 / .NET 7: Introduced static abstract members in interfaces, enabling static polymorphism.

Practical Examples of Interfaces

1. Payment Processing System

public interface IPaymentProcessor
{
    bool ProcessPayment(decimal amount);
}

public class PaypalProcessor : IPaymentProcessor
{
    public bool ProcessPayment(decimal amount)
    {
        Console.WriteLine("Processing PayPal payment");
        return true;
    }
}

public class StripeProcessor : IPaymentProcessor
{
    public bool ProcessPayment(decimal amount)
    {
        Console.WriteLine("Processing Stripe payment");
        return true;
    }
}

2. Logger Interface with Multiple Implementations

public interface ILogger
{
    void LogInfo(string message);
    void LogError(string message);
}

public class ConsoleLogger : ILogger
{
    public void LogInfo(string message) => Console.WriteLine($"INFO: {message}");
    public void LogError(string message) => Console.WriteLine($"ERROR: {message}");
}

public class FileLogger : ILogger
{
    public void LogInfo(string message) { /* write to file */ }
    public void LogError(string message) { /* write to file */ }
}


logo

C#

Beginner 5 Hours

Interfaces in C#

Interfaces in C# are a fundamental part of the language's support for abstraction and polymorphism. They allow you to define contracts that classes or structs must implement, enabling flexible and loosely coupled designs. This document provides a thorough exploration of interfaces, including their syntax, purpose, advanced features, and practical use cases.

What is an Interface?

An interface is a reference type in C# that defines a set of method signatures, properties, events, or indexers without any implementation. It specifies what a class or struct should do but not how it does it.

By implementing an interface, a class or struct agrees to provide implementations for all members declared in the interface.

Basic Syntax of an Interface

public interface IShape
{
    double GetArea();
    double GetPerimeter();
}

Here, IShape declares two methods that any implementing class must define.

Implementing an Interface

public class Rectangle : IShape
{
    public double Width { get; set; }
    public double Height { get; set; }

    public double GetArea()
    {
        return Width * Height;
    }

    public double GetPerimeter()
    {
        return 2 * (Width + Height);
    }
}

Why Use Interfaces?

Interfaces provide several important benefits:

  • Abstraction: Define a contract without implementation.
  • Polymorphism: Different classes can be treated uniformly if they implement the same interface.
  • Decoupling: Code depending on interfaces rather than concrete classes is more flexible and testable.
  • Multiple Inheritance: C# doesn’t support multiple inheritance of classes, but a class can implement multiple interfaces.

Interface Members

Interfaces can declare several types of members, all of which are implicitly public and abstract (no implementation, unless using default interface methods - see below):

  • Methods
  • Properties
  • Events
  • Indexers

Example with Properties and Events

public interface INotifier
{
    event EventHandler NotificationReceived;

    string Status { get; set; }

    void Notify(string message);
}

Implementing Multiple Interfaces

A class or struct can implement more than one interface, allowing a flexible composition of behaviors.

public interface IFlyable
{
    void Fly();
}

public interface ISwimmable
{
    void Swim();
}

public class Duck : IFlyable, ISwimmable
{
    public void Fly()
    {
        Console.WriteLine("Flying");
    }

    public void Swim()
    {
        Console.WriteLine("Swimming");
    }
}

Explicit Interface Implementation

When a class implements multiple interfaces that have members with the same signature, or when you want to hide interface methods from the public API, you can implement the interface explicitly.

Example

public interface IFirst
{
    void DoWork();
}

public interface ISecond
{
    void DoWork();
}

public class Worker : IFirst, ISecond
{
    void IFirst.DoWork()
    {
        Console.WriteLine("First DoWork");
    }

    void ISecond.DoWork()
    {
        Console.WriteLine("Second DoWork");
    }
}

Usage:

Worker worker = new Worker();

((IFirst)worker).DoWork();   // Output: First DoWork
((ISecond)worker).DoWork();  // Output: Second DoWork

Default Interface Methods (C# 8.0+)

Starting from C# 8.0, interfaces can provide default implementations for members. This allows interface authors to add new members without breaking existing implementations.

public interface ILogger
{
    void Log(string message)
    {
        Console.WriteLine($"Log: {message}");
    }
}

Classes implementing ILogger can override Log but aren’t required to.

Example

public class ConsoleLogger : ILogger
{
    // Uses default Log implementation
}

public class FileLogger : ILogger
{
    public void Log(string message)
    {
        // Custom file logging logic
    }
}

Interface Inheritance

Interfaces can inherit from one or more other interfaces, enabling hierarchical contracts.

public interface IMovable
{
    void Move();
}

public interface IAnimal : IMovable
{
    void Eat();
}

public class Dog : IAnimal
{
    public void Move()
    {
        Console.WriteLine("Dog moves");
    }

    public void Eat()
    {
        Console.WriteLine("Dog eats");
    }
}

Interfaces vs Abstract Classes

Aspect Interface Abstract Class
Purpose Defines a contract with no implementation (except default methods) Defines a partial implementation
Multiple Inheritance Supported (a class can implement multiple interfaces) Not supported (only single inheritance)
Members Methods, properties, events, indexers Can contain fields, constructors, full method implementations
Access Modifiers Members are implicitly public Members can have access modifiers
Versioning Default implementations from C# 8.0 allow better versioning Can add new methods without breaking derived classes

Interface Variables and Polymorphism

Variables of an interface type can reference any object that implements the interface, enabling polymorphism.

public interface IWorker
{
    void Work();
}

public class Developer : IWorker
{
    public void Work()
    {
        Console.WriteLine("Writing code");
    }
}

public class Manager : IWorker
{
    public void Work()
    {
        Console.WriteLine("Managing team");
    }
}

IWorker worker = new Developer();
worker.Work(); // Output: Writing code

worker = new Manager();
worker.Work(); // Output: Managing team

Interface Segregation Principle (ISP)

One of the SOLID design principles states that no client should be forced to depend on methods it does not use. Interfaces enable this by breaking large interfaces into smaller, more specific ones.

Example

public interface IPrinter
{
    void Print();
}

public interface IScanner
{
    void Scan();
}

public class MultiFunctionPrinter : IPrinter, IScanner
{
    public void Print() { /* ... */ }
    public void Scan() { /* ... */ }
}

public class SimplePrinter : IPrinter
{
    public void Print() { /* ... */ }
}

Interfaces with Generics

Interfaces can be generic, allowing type parameters to specify flexible contracts.

public interface IRepository
{
    void Add(T item);
    T Get(int id);
}
public class UserRepository : IRepository
{
    public void Add(User item) { /* ... */ }
    public User Get(int id) { /* ... */ return new User(); }
}

Interfaces and Dependency Injection (DI)

Interfaces are widely used in DI frameworks to inject dependencies, enhancing testability and modularity.

// Define an interface
public interface IMessageService
{
    void SendMessage(string message);
}

// Implement the interface
public class EmailService : IMessageService
{
    public void SendMessage(string message)
    {
        Console.WriteLine("Sending email: " + message);
    }
}

// Consumer class
public class Notification
{
    private readonly IMessageService _messageService;

    public Notification(IMessageService messageService)
    {
        _messageService = messageService;
    }

    public void Notify(string message)
    {
        _messageService.SendMessage(message);
    }
}

Explicit Interface Implementation and Access Control

Explicit interface implementation allows methods to be accessible only via interface references, hiding them from the public API of the class.

public interface ILogger
{
    void Log(string message);
}

public class SilentLogger : ILogger
{
    void ILogger.Log(string message)
    {
        // Does nothing (silently ignores logs)
    }
}

SilentLogger logger = new SilentLogger();
// logger.Log("Hi"); // Error: not accessible
((ILogger)logger).Log("Hi"); // Works fine

Interfaces with Default Implementations (Example)

public interface IShape
{
    double GetArea();

    double GetPerimeter()
    {
        return 0; // Default implementation returns zero
    }
}

Limitations of Interfaces

  • Interfaces cannot contain fields (state).
  • Prior to C# 8, interfaces could not contain any implementation.
  • Interfaces can become difficult to maintain if overused or poorly designed.
  • Versioning can be challenging without default implementations.

Interface Evolution in C#

Interfaces have evolved significantly:

  • Early C#: Interfaces only declared signatures.
  • C# 8: Added default interface methods, static methods in interfaces.
  • C# 11 / .NET 7: Introduced static abstract members in interfaces, enabling static polymorphism.

Practical Examples of Interfaces

1. Payment Processing System

public interface IPaymentProcessor
{
    bool ProcessPayment(decimal amount);
}

public class PaypalProcessor : IPaymentProcessor
{
    public bool ProcessPayment(decimal amount)
    {
        Console.WriteLine("Processing PayPal payment");
        return true;
    }
}

public class StripeProcessor : IPaymentProcessor
{
    public bool ProcessPayment(decimal amount)
    {
        Console.WriteLine("Processing Stripe payment");
        return true;
    }
}

2. Logger Interface with Multiple Implementations

public interface ILogger
{
    void LogInfo(string message);
    void LogError(string message);
}

public class ConsoleLogger : ILogger
{
    public void LogInfo(string message) => Console.WriteLine($"INFO: {message}");
    public void LogError(string message) => Console.WriteLine($"ERROR: {message}");
}

public class FileLogger : ILogger
{
    public void LogInfo(string message) { /* write to file */ }
    public void LogError(string message) { /* write to file */ }
}


Related Tutorials

Frequently Asked Questions for C#

C# is much easier to learn than C++. C# is a simpler, high-level-of-abstraction language, while C++ is a low-level language with a higher learning curve.

C# outshines Python when it comes to runtime performance. As a compiled language, C# code is converted to machine code, which can be executed more efficiently by the processor. This results in faster execution times and better performance, especially in resource-intensive tasks.

Python and JavaScript programmers also earn high salaries, ranking #3 and #4 in compensation. 
C# is the highest-paid programming language but has less demand than Python, JavaScript, and Java.

No. Microsoft has invested substantially in ensuring that C# is the dominant language today, spending two billion dollars on marketing and attempting to convince developers to embrace this new platform, which is also based on the.NET foundation.

C# is primarily used on the Windows .NET framework, although it can be applied to an open source platform. This highly versatile programming language is an object-oriented programming language (OOP) and comparably new to the game, yet a reliable crowd pleaser.


You can’t be able to become Master of C# in 3 months since it has many concepts to learn and implement. NOTE: no one can become master in particular programming language. Everyday they introducing new concepts we need to get practice on it which practically somewhat tough.

C-Sharp is one of the most widely used languages for creating system backend.It's because of its incredible features, such as Windows server automation. Apart from that, it's fantastic because it runs codes quite quickly. It can also be used to create CLI applications and game creation.

Easy to learn and use: C# is simpler than Java due to its use of fewer keywords and usually shorter lines of code. Hence, it is easier to learn to code in C# compared to Java. Flexible Data Types: C# provides more flexibility in defining data types than Java.

Four steps of code compilation in C# include : 
  • Source code compilation in managed code.
  • Newly created code is clubbed with assembly code.
  • The Common Language Runtime (CLR) is loaded.
  • Assembly execution is done through CLR.

The C# language is also easy to learn because by learning a small subset of the language you can immediately start to write useful code. More advanced features can be learnt as you become more proficient, but you are not forced to learn them to get up and running. C# is very good at encapsulating complexity.


The decision to opt for C# or Node. js largely hinges on the specific requirements of your project. If you're developing a CPU-intensive, enterprise-level application where stability and comprehensive tooling are crucial, C# might be your best bet.


Among other languages, C# is gaining huge popularity for developing web-based applications. Its core concepts help build an interactive environment and provide functionalities that the dynamic web platform requires. Most aspiring full-stack developers choose this versatile language.

The C# programming language was designed by Anders Hejlsberg from Microsoft in 2000 and was later approved as an international standard by Ecma (ECMA-334) in 2002 and ISO/IEC (ISO/IEC 23270 and 20619) in 2003. Microsoft introduced C# along with .NET Framework and Visual Studio, both of which were closed-source. 

C# outshines Python when it comes to runtime performance. As a compiled language, C# code is converted to machine code, which can be executed more efficiently by the processor. This results in faster execution times and better performance, especially in resource-intensive tasks.

Yes, C# is used by many large organizations, start-ups and beginners alike. It takes some of the useful features of C and adds syntax to save time and effort. Although C# is based on C, you can learn it without any knowledge of C β€” in fact, this course is perfect for those with no coding experience at all!

C# is a very mature language that evolved significantly over the years.
The C# language is one of the top 5 most popular programming languages and .NET is the most loved software development framework in the world.
TIOBE Index predicts C# as 2023 'Language of the Year' close to overtake Java in popularity.

Generally, the C# language is not limited to the Windows operating system. In a sense, however, it is limited to Microsoft software. C# language "belongs" to Microsoft, it is developed by Microsoft and it is Microsoft that provides the runtime environment required for the operation of programs written in C#.

C# (pronounced "C sharp") is called so because the "#" symbol is often referred to as "sharp." The name was chosen by Microsoft when they developed the language. It's a play on words related to musical notation where "C#" represents the musical note C sharp.

Dennis MacAlistair Ritchie (September 9, 1941 – c. October 12, 2011) was an American computer scientist. He created the C programming language and, with long-time colleague Ken Thompson, the Unix operating system and B language.

C# is part of .NET, a free and open source development platform for building apps that run on Windows, macOS, Linux, iOS, and Android. There's an active community answering questions, producing samples, writing tutorials, authoring books, and more.


line

Copyrights © 2024 letsupdateskills All rights reserved