Object-Oriented Programming (OOP)


Table of Contents
  1. What is Object-Oriented Programming (OOP) in Java?
  2. Main Features of Object-Oriented Programming (OOP)
  3. What is encapsulation in Java?
  4. Explain inheritance in Java with an example
  5. What is polymorphism in Java?
  6. What is abstraction, and how is it achieved in Java?
  7. Explain the difference between an abstract class and an interface.
  8. How does Java implement multiple inheritance?
  9. Can a class be declared as both `abstract` and `final`?
  10. What is a constructor, and what are its types?
  11. What is the role of the `super` keyword in Java?
  12. Explain the concept of method overloading and method overriding
  13. Can we overload a static method in Java?
  14. Can you override a static method in Java?
  15. Can we prevent overriding a method without using the final modifier?
  16. Can we override a private method in Java?
  17. Can We Change the Return Type of a Method to a Subclass While Overriding?
  18. What is an inner class, and why is it used?
  19. What is the difference between `this` and `super` keywords?
  20. How Do You Call a Superclass Version of an Overriding Method in a Subclass?
  21. How do you achieve immutability in Java?
  22. What is a sealed class in Java?
  23. What is the method of hiding in Java?
  24. Can we make a class both final and abstract at the same time?
  25. Can a class extend more than one class in Java?
  26. Can an interface extend another interface?
  27. What is the `instanceof` keyword?
  28. What is the purpose of the `default` methods in interfaces?
  29. Explain the use of functional interfaces in Java 8.
  30. How does Java handle object cloning?
  31. What is composition in OOP?
  32. Explain dependency injection.
  33. What are default constructors?
  34. Can you explain the concept of method hiding?
  35. What are the SOLID principles
  36. What is the difference between aggregation and composition?
  37. What is the difference between method overloading and method overriding?
  38. What happens if you override a method and change the return type to a different type, which is not a subclass of the original return type?
  39. Can a constructor be inherited?
  40. Can we call a constructor from a static method in Java?

What is Object-Oriented Programming (OOP) in Java?
  • OOP is a programming paradigm that organizes software design around objects, which are instances of classes.
  • An object is an instance of a class, and a class serves as a blueprint defining properties (fields) and behaviors (methods).
  • It is based on four fundamental principles: Encapsulation, Inheritance, Polymorphism, and Abstraction.
  • OOP promotes code reusability, modularity, scalability, and easier maintenance.
  • Java fully supports OOP concepts, making it a widely used language for OOP-based software development.
  • Detail of OOPs

    Main Features of Object-Oriented Programming (OOP)

    Object-Oriented Programming (OOP) is a programming paradigm that organizes software design around data, or objects, rather than functions and logic. The key features of OOP help make the development process more modular, reusable, and easier to maintain.

    1. Encapsulation:

    Encapsulation is the concept of bundling the data (variables) and the methods (functions) that operate on the data into a single unit, called a class. It also helps in restricting access to certain components of the object to protect the data from unauthorized access and modification.

    2. Abstraction:

    Abstraction is the concept of hiding the complex implementation details and showing only the essential features of an object. It allows you to focus on what an object does rather than how it does it.

    3. Inheritance:

    Inheritance is a mechanism where a new class inherits the properties and behaviors (methods) of an existing class. It helps in reusing code, extending functionality, and creating a hierarchical class structure.

    4. Polymorphism:

    Polymorphism means "many forms." It allows objects to be treated as instances of their parent class, but the actual method that gets executed is determined by the object's actual class (at runtime in the case of dynamic polymorphism).

    5. Object:

    An object is an instance of a class. It holds real values for the variables defined in the class and can perform actions (methods) defined by the class. Objects are the basic units of OOP.

    Summary:

    What is encapsulation in Java?
    Data Hiding - Encapsulation


    Explain inheritance in Java with an example.
    Inheritance

    What is polymorphism in Java?
    Polymorphism

    What is abstraction, and how is it achieved in Java?
    Data Hiding - Abstraction

    Explain the difference between an abstract class and an interface.
    Feature Abstract Class Interface
    Definition A class that can have both abstract methods (without a body) and concrete methods (with a body). A blueprint that only contains abstract methods (until Java 8, after which default and static methods were introduced).
    Usage Used to represent a base class that provides partial implementation for subclasses. Used to define a contract that implementing classes must adhere to.
    Inheritance A class can extend only one abstract class (single inheritance). A class can implement multiple interfaces (multiple inheritance).
    Access Modifiers Abstract classes can have any access modifier for methods and fields (e.g., private, protected, public). All methods in an interface are public by default.
    Fields Can have instance variables and static variables. Fields are implicitly public, static, and final (constants).
    Constructor Can have a constructor to initialize fields. Cannot have a constructor.
    Example
              abstract class Animal {
                  abstract void makeSound();
                  void eat() {
                      System.out.println("Eating...");
                  }
              }
              
              interface Animal {
                  void makeSound();
                  default void eat() {
                      System.out.println("Eating...");
                  }
              }
              
    Note:


    How does Java implement multiple inheritance?

    Answer: Java implements multiple inheritance through interfaces.
    A class can implement multiple interfaces, thus inheriting behavior from all of them.
    This avoids the "diamond problem" caused by ambiguity in multiple inheritance.

          interface InterfaceA {
              void methodA();
          }
          
          interface InterfaceB {
              void methodB();
          }
          
          class MultiInheritanceClass implements InterfaceA, InterfaceB {
              public void methodA() {
                  System.out.println("Method A from InterfaceA");
              }
          
              public void methodB() {
                  System.out.println("Method B from InterfaceB");
              }
          }
          
          public class Test {
              public static void main(String[] args) {
                  MultiInheritanceClass obj = new MultiInheritanceClass();
                  obj.methodA();
                  obj.methodB();
              }
          }
            

    Can a class be declared as both `abstract` and `final`?

    Answer: No, a class cannot be both abstract and final because an abstract class is meant to be extended, whereas a final class cannot be extended.


    What is a constructor, and what are its types?

    Answer: A constructor is a special method used to initialize objects. Types:


    What is the role of the `super` keyword in Java?

    Answer: The super keyword is used to refer to the immediate parent class. It can be used to call the parent class constructor or methods.

    class Animal {
          void eat() {
              System.out.println("Animal eats.");
          }
      }
      
      class Dog extends Animal {
          void eat() {
              super.eat();
              System.out.println("Dog eats.");
          }
      }
      

    Explain the concept of method overloading and method overriding.

    Method Overloading: This occurs when multiple methods in the same class share the same name but have different parameter lists (number, type, or order of parameters). It is a compile-time polymorphism.

    Key Points:

          class Calculator {
              // Overloaded methods
              int add(int a, int b) {
                  return a + b;
              }
          
              double add(double a, double b) {
                  return a + b;
              }
          
              int add(int a, int b, int c) {
                  return a + b + c;
              }
          }
          
          public class OverloadingExample {
              public static void main(String[] args) {
                  Calculator calc = new Calculator();
                  System.out.println(calc.add(2, 3));        // Calls int add(int, int)
                  System.out.println(calc.add(2.5, 3.5));   // Calls double add(double, double)
                  System.out.println(calc.add(1, 2, 3));    // Calls int add(int, int, int)
              }
          }
            

    Method Overriding: This occurs when a subclass provides a specific implementation for a method already defined in its parent class. It is runtime polymorphism.

    Key Points:

          class Parent {
              void display() {
                  System.out.println("Display from Parent");
              }
          }
          
          class Child extends Parent {
              @Override
              void display() {
                  System.out.println("Display from Child");
              }
          }
          
          public class OverridingExample {
              public static void main(String[] args) {
                  Parent obj = new Child(); // Upcasting
                  obj.display(); // Calls Child's overridden method
              }
          }
            

    Tricky Points:


    Can we overload a static method in Java?

    Yes, you can overload a static method in Java. You can declare as many static methods of the same name as you wish provided all of them have different method signatures.

          class Calculator {
              // Static method to add two integers
              static int add(int a, int b) {
                  return a + b;
              }
          
              // Static method to add three integers
              static int add(int a, int b, int c) {
                  return a + b + c;
              }
          
              // Static method to add two double values
              static double add(double a, double b) {
                  return a + b;
              }
          }
          
          public class StaticMethodOverloadingDemo {
              public static void main(String[] args) {
                  // Calling overloaded static methods
                  System.out.println(Calculator.add(10, 20));           // Calls method with two integers
                  System.out.println(Calculator.add(10, 20, 30));       // Calls method with three integers
                  System.out.println(Calculator.add(10.5, 20.5));       // Calls method with two doubles
              }
          }
              
    Output:
          30
          60
          31.0
        


    Can you override a static method in Java?

    No, static methods cannot be overridden in Java. Instead, if a static method is defined in both the parent and child classes with the same name and parameters, it is known as method hiding. The method called depends on the reference type, not the object type.

          class Parent {
              static void staticMethod() {
                  System.out.println("Static method in Parent");
              }
          }
          
          class Child extends Parent {
              static void staticMethod() {
                  System.out.println("Static method in Child");
              }
          }
          
          public class StaticMethodTest {
              public static void main(String[] args) {
                  Parent parent = new Parent();
                  Parent childAsParent = new Child();
                  Child child = new Child();
                  
                  parent.staticMethod();           // Outputs: Static method in Parent
                  childAsParent.staticMethod();    // Outputs: Static method in Parent (method hiding)
                  child.staticMethod();            // Outputs: Static method in Child
              }
          }
            

    Key Points:


    Can we prevent overriding a method without using the final modifier?

    Yes, you can prevent the method overriding in Java without using the final modifier. In fact, there are several ways to accomplish it e.g. you can mark the method as private or static, those cannot be overridden.


    Can we override a private method in Java?

    No, you cannot. Since the private method is only accessible and visible inside the class they are declared, it's not possible to override them in subclasses. Though, you can override them inside the inner class as they are accessible there.


    Can We Change the Return Type of a Method to a Subclass While Overriding?

    Yes, in Java, we can change the return type of a method to a subclass while overriding the method. This feature is called covariant return type.

    Covariant Return Type:

    Covariant return type allows a subclass to override a method and return a type that is a subclass of the original method's return type in the parent class. It is a feature introduced in Java 5 that enhances flexibility and reusability while preserving method signatures and behavior.

    Example of Covariant Return Type in Java:
    class Animal {
      // Method returning an Animal object
      Animal getAnimal() {
          return new Animal();
      }
    }
    
    class Dog extends Animal {
      // Overriding the method with a covariant return type (returning Dog)
      @Override
      Dog getAnimal() {
          return new Dog();
      }
    }
    
    public class CovariantReturnTypeDemo {
      public static void main(String[] args) {
          Animal animal = new Animal();
          Animal anotherAnimal = animal.getAnimal(); // Returns an Animal object
    
          Dog dog = new Dog();
          Dog anotherDog = dog.getAnimal(); // Returns a Dog object, covariant return type
      }
    }
      
    Output:
    No output, but the types are correct:
    - animal.getAnimal() returns Animal
    - dog.getAnimal() returns Dog
      
    Explanation of the Example: Benefits of Covariant Return Type: Key Restrictions:

    What is an inner class, and why is it used?

    Answer: An inner class in Java is a class that is defined within another class. Inner classes are used to logically group classes that are only used in one place, improve encapsulation, and make the code more readable and maintainable.

    Types of Inner Classes:

    Example: Member Inner Class

          class OuterClass {
              private String message = "Hello from Outer Class!";
              
              class InnerClass {
                  void displayMessage() {
                      System.out.println(message); // Accessing private member of the outer class
                  }
              }
          }
          
          public class InnerClassExample {
              public static void main(String[] args) {
                  OuterClass outer = new OuterClass();
                  OuterClass.InnerClass inner = outer.new InnerClass();
                  inner.displayMessage();
              }
          }
            

    Why Use Inner Classes?


    What is the difference between `this` and `super` keywords?

    Answer:


    How Do You Call a Superclass Version of an Overriding Method in a Subclass?

    In Java, when a method in a subclass overrides a method in its superclass, you can call the superclass version of the method using the super keyword. The super keyword allows access to the superclass's methods and constructors.

    Example of Calling a Superclass Method from a Subclass:
    class Animal {
      // Method in the superclass
      void sound() {
          System.out.println("Animal makes a sound");
      }
    }
    
    class Dog extends Animal {
      // Overriding the sound method
      @Override
      void sound() {
          System.out.println("Dog barks");
          
          // Calling the superclass version of the sound method
          super.sound();
      }
    }
    
    public class SuperclassMethodCallDemo {
      public static void main(String[] args) {
          Dog dog = new Dog();
          dog.sound();  // Calls the overridden method in Dog class
      }
    }
      
    Output:
    Dog barks
    Animal makes a sound
      
    Limitations:

    How do you achieve immutability in Java?

    Answer: Create a class with:

    final class Immutable {
          private final String name;
      
          Immutable(String name) {
              this.name = name;
          }
      
          public String getName() {
              return name;
          }
      }
      

    What is a sealed class in Java?

    Answer: A sealed class in Java restricts the inheritance of the class to a specific set of classes. It is a way to restrict the subclassing of a class, ensuring that only a designated set of classes can inherit it. Sealed classes are introduced in Java 16.

    Key Features:

    Example:

          sealed class Shape permits Circle, Rectangle {
              double area;
          
              Shape(double area) {
                  this.area = area;
              }
          
              abstract double calculateArea();
          }
          
          final class Circle extends Shape {
              Circle(double area) {
                  super(area);
              }
          
              @Override
              double calculateArea() {
                  return Math.PI * area * area;
              }
          }
          
          final class Rectangle extends Shape {
              double length, width;
          
              Rectangle(double area, double length, double width) {
                  super(area);
                  this.length = length;
                  this.width = width;
              }
          
              @Override
              double calculateArea() {
                  return length * width;
              }
          }
          
          public class SealedClassExample {
              public static void main(String[] args) {
                  Shape circle = new Circle(10.0);
                  Shape rectangle = new Rectangle(20.0, 5.0, 4.0);
                  
                  System.out.println("Circle Area: " + circle.calculateArea());
                  System.out.println("Rectangle Area: " + rectangle.calculateArea());
              }
          }
            

    Why Use Sealed Classes?


    What is the method of hiding in Java?

    When you declare two static methods with the same name and signature in both superclass and subclass then they hide each other i.e. a call to the method in the subclass will call the static method declared in that class and a call to the same method is superclass is resolved to the static method declared in the super-class.


    Can we make a class both final and abstract at the same time?

    No, you cannot apply both final and abstract keywords to the class at the same time because they are exactly the opposite of each other. A final class in Java cannot be extended and you cannot use an abstract class without extending and making it a concrete class. As per Java specification, the compiler will throw an error if you try to make a class abstract and final at the same time.


    Can a class extend more than one class in Java?

    No, a class can only extend another class because Java doesn't support multiple inheritances but yes, it can implement multiple interfaces.


    Can an interface extend another interface?

    Answer: Yes, an interface can extend another interface. This allows a child interface to inherit methods from the parent interface.


    What is the `instanceof` keyword?

    Answer: The instanceof keyword is used to test whether an object is an instance of a specific class or subclass.


    What is the purpose of the `default` methods in interfaces?

    Answer: Default methods, introduced in Java 8, allow adding new methods to interfaces without breaking existing implementations.


    Explain the use of functional interfaces in Java 8.

    Answer: A functional interface is an interface with a single abstract method. It is used in lambda expressions and method references. Example: Runnable, Predicate.


    How does Java handle object cloning?

    Answer: Java provides cloning using the clone() method of the Object class. A class must implement Cloneable to use it.


    What is composition in OOP?

    Answer: Composition is a design principle where a class contains objects of other classes to achieve reusability.


    Explain dependency injection.

    Answer: Dependency Injection is a design pattern where objects are provided their dependencies instead of creating them internally, promoting loose coupling.


    What are default constructors?

    Answer: A default constructor is provided by Java if no constructor is explicitly defined. It initializes objects with default values.


    Can you explain the concept of method hiding?

    Answer: Method hiding occurs when a static method in a subclass has the same name and signature as one in its superclass. The method is hidden, not overridden.


    What are the SOLID principles?

    Answer: SOLID principles are design principles for maintainable software:


    What is the difference between aggregation and composition?

    Answer:


    What is the difference between method overloading and method overriding?

    Method overloading occurs when you have multiple methods with the same name but different parameter lists (either by number, type, or both) in the same class. It is a form of compile-time polymorphism. Method overriding occurs when a subclass provides a specific implementation of a method that is already defined in its superclass, and it is a form of runtime polymorphism.


    What happens if you override a method and change the return type to a different type, which is not a subclass of the original return type?

    Java does not allow overriding a method with a return type that is not a subclass of the original return type. The return type in the overriding method must either be the same as the return type in the superclass method or a subtype of it (covariant return type). If you attempt to do so, a compilation error will occur.


    Can a constructor be inherited?

    No, constructors are not inherited in Java. Although a subclass can call a superclass constructor using the super keyword, it cannot directly inherit the constructor from the superclass. The subclass can only inherit the behavior of the superclass via method inheritance, not constructors.


    Can we call a constructor from a static method in Java?

    Yes, you can call a constructor from a static method in Java, but you cannot directly call a constructor using this() in a static context (since this refers to an instance). However, you can create an object inside a static method using new and invoke the constructor to initialize the object.