Showing posts with label Design Pattern. Show all posts
Showing posts with label Design Pattern. Show all posts

What is difference in Inversion of Control and Dependency Injection

Inversion of Control (IoC) and Dependency Injection (DI) are related concepts in software design and architecture, but they are not the same thing. Here are the differences between the two:

Definition: Inversion of Control (IoC) is a pattern or principle that advocates for decoupling components and their dependencies by inverting the control of the flow of a program from the component to a container or framework. Dependency Injection (DI) is a specific implementation of IoC that involves providing the dependencies of a class from an external source, typically a container or framework.

Relationship: Dependency Injection is a specific technique that is used to implement Inversion of Control. It's one way to achieve IoC, but not the only way.

Scope: Inversion of Control is a broad concept that can be applied to many areas of software design and architecture, such as event-driven programming, template method pattern, and others. Dependency Injection, on the other hand, is more focused and specific to managing the dependencies of a component.

Purpose: Inversion of Control is aimed at improving the modularity, maintainability, testability, and extensibility of a software system by reducing the coupling between components. Dependency Injection, as a form of IoC, specifically aims to reduce coupling by managing dependencies of a component and making it easier to change its behavior or add new features.

In summary, Dependency Injection is a specific implementation of the more general principle of Inversion of Control. Dependency Injection provides an easy and efficient way to implement IoC by enabling the creation of loosely coupled components with their dependencies provided from an external source, which makes the software system more modular, maintainable, testable, and extensible.

Example of Implementing dependency Injection -

here's a simple example of how to implement Dependency Injection in C# using the constructor injection technique:

Let's say we have a Customer class that needs to use a CustomerRepository class to retrieve and save customer data from a database. Instead of instantiating the CustomerRepository class inside the Customer class, we can use Dependency Injection to provide an instance of the CustomerRepository class from outside the Customer class. 

Here's how to do it:

public interface ICustomerRepository
{
    Customer GetCustomer(int id);
    void SaveCustomer(Customer customer);
}

public class CustomerRepository : ICustomerRepository
{
    public Customer GetCustomer(int id)
    {
        // implementation to retrieve customer data from a database
    }

    public void SaveCustomer(Customer customer)
    {
        // implementation to save customer data to a database
    }
}

public class Customer
{
    private readonly ICustomerRepository _repository;

    public Customer(ICustomerRepository repository)
    {
        _repository = repository;
    }

    public int Id { get; set; }
    public string Name { get; set; }

    public void Save()
    {
        _repository.SaveCustomer(this);
    }

    public static Customer Get(int id, ICustomerRepository repository)
    {
        var customer = repository.GetCustomer(id);
        return new Customer(repository)
        {
            Id = customer.Id,
            Name = customer.Name
        };
    }
}

In this example, we define an interface ICustomerRepository and a class CustomerRepository that implements the interface. The Customer class has a constructor that takes an ICustomerRepository object as a parameter, which is assigned to a private field _repository. The Save method of the Customer class uses the _repository object to save customer data to a database.

In addition, we have a static method Get that takes an ICustomerRepository object as a parameter and uses it to retrieve a customer from the database. This allows us to create a Customer object with a specific ICustomerRepository instance without requiring a default constructor.

To use this setup, we can create an instance of the CustomerRepository class and pass it to the Customer class constructor as follows:

ICustomerRepository repository = new CustomerRepository();
Customer customer = new Customer(repository);
customer.Name = "John";
customer.Save();

In this way, we can achieve dependency injection and have the flexibility to change the implementation of the ICustomerRepository interface without changing the Customer class code.

What is Difference in Static Class and Singleton

A static class and a singleton are two different design patterns in C# that are used for different purposes.

A static class is a class that can only contain static members (methods, properties, fields). It cannot be instantiated, and all its members are accessible through the class name itself. A static class is typically used to group related utility methods or constants that do not require any state or instance-specific behavior. 

Here's an example of a static class:

public static class Calculator
{
    public static int Add(int x, int y)
    {
        return x + y;
    }

    public static int Multiply(int x, int y)
    {
        return x * y;
    }
}

A singleton, on the other hand, is a class that can have only one instance throughout the lifetime of the application. It typically has a private constructor to prevent direct instantiation and provides a static method or property to access the singleton instance. A singleton is often used to represent a global state or to provide a centralized point of access to a resource or service. 

Here's an example of a singleton:

public class Logger
{
    private static Logger instance;
    private static readonly object lockObject = new object();

    private Logger()
    {
        // Private constructor to prevent direct instantiation
    }

    public static Logger Instance
    {
        get
        {
            // Double-checked locking to ensure thread-safety
            if (instance == null)
            {
                lock (lockObject)
                {
                    if (instance == null)
                    {
                        instance = new Logger();
                    }
                }
            }

            return instance;
        }
    }

    public void Log(string message)
    {
        // Logging implementation
    }
}

The main difference between a static class and a singleton is that a static class cannot be instantiated, whereas a singleton can have one instance. A static class is typically used to group related utility methods or constants, while a singleton is often used to represent a global state or to provide a centralized point of access to a resource or service.