SOLID Principals - Interface Segregation Principle (ISP)

Interface Segregation Principle (ISP) in Ruby: A Practical Guide

The Interface Segregation Principle (ISP) is one of the SOLID principles aimed at guiding object-oriented design. It emphasizes that a class should not be forced to implement interfaces it does not use. In other words, clients should not be forced to depend on interfaces they do not use. In this article, we’ll explore the Interface Segregation Principle through multiple Ruby examples.

Understanding Interface Segregation Principle

ISP suggests that it is better to have many small, specific interfaces tailored for individual classes than a large, general-purpose interface that forces a class to implement methods it doesn’t need. By adhering to ISP, we create more maintainable, flexible, and understandable code.

Example 1: Violation of ISP

Consider a scenario where we have a Worker class that performs both regular work and reports progress:

class Worker
  def do_work
    # Code to perform general work
  end

  def report_progress
    # Code to report progress
  end
end

In this example, the Worker class violates ISP because clients that only need to use the do_work method are forced to deal with the unnecessary report_progress method.

Refactoring for ISP

To adhere to ISP, we can split the interface into two separate interfaces, each catering to a specific set of responsibilities:

module Workable
  def do_work
    # Code to perform general work
  end
end

module ProgressReportable
  def report_progress
    # Code to report progress
  end
end

Now, classes can include the specific modules they need:

class Worker
  include Workable
end

class ProgressReporter
  include ProgressReportable
end

This adheres to the Interface Segregation Principle by allowing classes to include only the interfaces they require.

Example 2: Applying ISP to Services

Consider a service class that handles both user authentication and file management:

class FileManagementService
  def authenticate_user
    # Code to authenticate user
  end

  def upload_file(file)
    # Code to upload a file
  end
end

In this case, the FileManagementService class has a dual responsibility, violating ISP. Let’s refactor this by creating separate services for authentication and file management:

class AuthenticationService
  def authenticate_user
    # Code to authenticate user
  end
end

class FileManagementService
  def upload_file(file)
    # Code to upload a file
  end
end

Now, we have two classes, each with a single responsibility. This adheres to ISP by allowing clients to depend on specific interfaces without being burdened by unnecessary methods.

Conclusion

The Interface Segregation Principle is about creating focused and specific interfaces tailored to the needs of individual classes. By adhering to ISP, developers can build more modular and maintainable systems. The examples provided demonstrate how to refactor code to follow ISP principles in the context of Ruby, fostering a design that enhances flexibility and reduces dependencies between classes.

Related Posts