What are the core principles of object-oriented programming?

Today, programming adopts various paradigms for writing and packaging code. These paradigms include functional, procedural, logic, object-oriented, access-oriented, and constraint-based programming. Pokkunuri(1) notes that there is no universal consensus on the “best” paradigm, as each has its own advantages and trade-offs.

Object-oriented programming (OOP) focuses on solving real-world problems using objects. Objects are abstract representations of real-world entities, and the code (programs or blocks of code) is structured to model and solve these problems.

OOP vs. Other Paradigms

OOP differs from other programming paradigms, such as functional programming. In functional programming, functions are defined in a sequence and applied to data. OOP, on the other hand, focuses on objects and their behaviors or interactions with other objects, according to Tusa(2).

OOP is based on four core principles:

  1. Abstraction
  2. Encapsulation
  3. Polymorphism
  4. Inheritance

These principles help organize and structure programs in a way that mimics real-world entities and their behaviors.

Objects: Attributes and Methods

In an OOP context, objects have two key components:

  • Attributes: These describe the state or properties of the object.
  • Methods: These define the behaviors or operations that the object can perform.

For example, consider an object representing a person:

const person = {
  // Private attributes and methods
  _firstName: "Jack",
  _lastName: "Jones",
  
  // Getter method for first name
  get firstName() {
    return this._firstName;
  },
  
  // Setter method for first name
  set firstName(value) {
    this._firstName = value;
  }
};

While the code above is in JavaScript, a completely different language Java has the same concept of an object written in different syntax. Java is strongly typed, meaning every variable has a data type and cannot be executed unless there is a type associated with an object.

To create an object in Java, a class is required. In JavaScript, objects are created by following the inherited prototypes of other objects. As such, JavaScript therefore mimics object-oriented programming by creating classes from factory functions but the underlying language does not inherently support the creation of classes out of the box.

Let’s look at a Java class to see how a class has attributes and methods in the example below.

package bmicalculator;

public class Person {

    private String name;
    private String surname;
    private Double weight;  // weight in kilograms
    private Double height;  // height in meters
    private Double bmi;

    public Person(String name, String surname, Double weight, Double height) {
        this.name = name;
        this.surname = surname;
        this.weight = weight;
        this.height = height;
        this.bmi = calculateBMI();  // Calculate BMI when the object is created
    }

    public Double calculateBMI() {
        if (weight != null && height != null && height > 0) {
            return weight / (height * height);
        }
        return 0.0; // Return 0.0 if input values are invalid
    }
}

What is the value of classes in creating new object instances?

Object-oriented programming relies on classes that are blueprints creating new object instances. The Class keyword in most object-oriented languages refers to the constructor that constructs new instances of objects.

Each object must be initialised or instantiated. Take the example above. If we want different people, with different names and to calculate the BMI this is what we would do

package bmicalculator;

public class Program {

    public static void main(String[] args) {
              
        Person p1 = new Person("John", "Doe", 90.4, 1.7);
        p1.getName();
        System.out.println("P1's name is: " + p1.getName());
        System.out.println("BMI p1: " + p1.getBMI());
        
        Person p2 = new Person("Jane", "Doe", 55.0, 1.7);
        System.out.println("P2's name is: " + p2.getName());
        System.out.println("P2's full name is: " + p2.getFullName());
        System.out.println("P2's name is now set to : " + p2.setName("Maria"));
        System.out.println("BMI p2: " + p2.getBMI());
        
                
    }
}

Now the class Person has performed its job of creating 2 new objects, which are abstractions of the real-life people Jane and John Doe, one with a weight of 80 kilos and height of 1.7 meters and the other with a weight of 65 kilos and 2.0 meters.

The two objects p1 and p2 which represent the individuals in the example, now can access the methods of the class and their BMI calculated.

What is abstraction in object-oriented programming?

Abstraction refers to simplifying complex systems by modeling real-world problems in code. In the Person class, for example, we abstract a real-life person by capturing their name, surname, weight, and height, as well as providing a method to calculate their BMI.

Abstraction helps break down a complex problem into manageable components. In this case, we abstract the problem of calculating BMI for different people using attributes and methods.

What is encapsulation in object-oriented programming?

Encapsulation refers to bundling the data (attributes) and methods that operate on the data within one unit (the class). In OOP, this often means making attributes private and providing public methods to access or modify them. This ensures that the internal workings of a class are hidden from outside manipulation, leading to modular and secure code.

The methods are public and can be accessed by other classes in the application software and code packages that use these classes to run programs.

As you will see in both the JavaScript and Java code blocks, the use of the this keyword. It highlights the encapsulation of an attribute or a method to “this” variable of person as described in JavaScript or “this” class of Person as described in Java.

JavaScript, unlike Java, has no concept of classes and uses prototypal inheritance from the Window Object to define classes.

In reality, the JavaScript object is created using the global Object class that is inherited from the Window Object like so: const person = new Object(); as you will see from the Mozilla documentation (3).

What is the hierarchy of methods in a Java class?

As you can see by the differences between Java and JavaScript, the object-oriented paradigm has differences in the way some of these core principles are implemented.

Pokkunuri, highlights these differences in his paper, discussing other languages that use the object-oriented paradigm, with examples from Smalltalk,(4) a purely object-oriented language with no primitives like strings and numbers, C++ and Objective-C which are object-oriented paradigms of the C language or object-Pascal, for Pascal.

A disambiguation note, Java bears no relationship with or to JavaScript. The languages and the use cases are completely different.

In a Class, not all methods are equal. There is a hierarchy of classes and depending on the programming language and the program architecture the hierarchy of classes may vary.

Java is a compiled language. For a better understanding of code packages and code compiling, the article on storing and accessing programming data in this section will provide some context.

For this article we will focus, from now onwards on the Java programming language and demonstrate the hierarchy of methods in the code compilation process in the Java language.

In Java, methods can have different levels of visibility (public, private, etc.) depending on their purpose. Here’s how different types of methods work:

The Main Method (Entry Point of Java Program) The main class in a program in Java class is a static method, it has no attributes and is the first to be called in the stack by the Java compiler, or JVM (Java Virtual Machine).

For the sake of the example, this package is called the BMI calculator. In the main class of this program, we initialise the class Person that has been created in a separate code block, the class with the constructor Person.

The main method is the entry point for a Java program. It’s a static method, meaning it can be called without creating an instance of the class.

package bmicalculator;

public class Program {

    public static void main(String[] args) {
        Person p1 = new Person("John", "Doe", 90.4, 1.7);
        System.out.println("P1's name is: " + p1.getName());
        System.out.println("BMI p1: " + p1.getBMI());
    }
}

It contains the core code on how the whole application will be run. There is only 1 main program in a package.

The constructor method The constructor method refers to all the classes in the package that create new objects.

package bmicalculator;

// attributes information hiding and encapsulation
public class Person {

    private String name;
    private String surname;
    private Double weight;
    private Double height;
    private Double bmi;

    // Constructor
    public Person(String name, String surname, Double weight, Double height) {
        this.name = name;
        this.surname = surname;
        this.weight = weight;
        this.height = height;
        this.bmi = calculateBMI();
    }
}

The difference between public and private methods A class can have both public and private methods. Public methods are available to other classes when called. A private method is applied only on the objects in the class it is created in.

Public methods are accessible from outside the class, while private methods can only be accessed within the class.

package bmicalculator;

// attributes information hiding and encapsulation
public class Person {

    private String name;
    private String surname;
    private Double weight;  // weight in kilograms
    private Double height;  // height in meters
    private Double bmi;

//instantiation with constructor
    public Person(String name, String surname, Double weight, Double height) {
        this.name = name;
        this.surname = surname;
        this.weight = weight;
        this.height = height;
        this.bmi = calculateBMI();  // Calculate BMI when the object is created
    }
//method made private as only this class requires it
    private Double calculateBMI() {
        if (weight != null && height != null && height > 0) {
            return weight / (height * height);
        }
        return 0.0; // Return 0.0 if input values are invalid
    }

}

The getter and setter methods Getter methods return the value of an object’s attribute, and setter methods modify it. These allow controlled access to an object’s state. To get the data in the class, getters return the state of the object while setters allow programmers to update or mutate the initial state of that data.

package bmicalculator;

// attributes information hiding and encapsulation
public class Person {

    private String name;
    private String surname;
    private Double weight;  // weight in kilograms
    private Double height;  // height in meters
    private Double bmi;

//instantiation with constructor
    public Person(String name, String surname, Double weight, Double height) {
        this.name = name;
        this.surname = surname;
        this.weight = weight;
        this.height = height;
        this.bmi = calculateBMI();  // Calculate BMI when the object is created
    }

//methods getters - read data
    public String getName() {
        return name;
    }

    //mutate state - setter methods change params
    public String setName(String updatedName) {
        return updatedName;
    }
    }
}

In this approach an object behaves like a primitive element, combining attributes and procedures. Data is transmitted by the method’s signature - which is the function name, its parameters and procedures. The function is then “called” or invoked and the program set in the class runs.

This, Pokkunuri says is in contrast with a data-structure-oriented-programming. In this paradigm, a single powerful data structure is used. In SQL, this is a relational database, in LISP a list, or in APL an array.

In object-oriented programming, each object is a modular and autonomous entity. It has attributes and methods that send messages through callable functions to corresponding objects.

So, in the main program, we can now access the name of the p2 abstraction of Jane Doe and update the name to Maria.

  Person p2 = new Person("Jane", "Doe", 55.0, 1.7);
        System.out.println("P2's name is: " + p2.getName());
        System.out.println("P2's full name is: " + p2.getFullName());
        System.out.println("P2's name is now set to : " + p2.setName("Maria"));
        System.out.println("BMI p2: " + p2.getBMI());

Static methods that call on classes not objects Static methods, Tusa elaborates(5), are called at run time and not at compile time. They are methods that are written without attributes as they are methods that invoke classes not objects.

The advantage of these methods is that they are invoked on a class and therefore, in this example, you can calculate the BMI of several individuals by invoking the Person class, not the p1, p2 objects that have been created by the class.

Static methods belong to the class itself rather than an instance of the class. They are called using the class name and do not require an object to be created. In this example, this static method, could also be written as a private static method in the class, extracting it from the class means the calculator function of BMI can be used in other parts of the code if for example, it is required to create a BMI homekit where the calculation is required as a standalone block of code to be reused.

public class BMICalculator {
    
    public static double calculateBMI(double weight, double height) {
        return weight / (height * height);
    }
}

The code we saw initially now can be modularised, with the calculator method taken out of the class and the constructor, getter and setter methods. It can stand alone and call the class to calculate the BMI of any number of people - which is the problem we are trying to solve with this abstraction.

Encapsulation, therefore, does not work here as the calculation of the BMI does not depend on the person but is a class of its own that needs to be applied to calculating a person’s weight. Therefore, in the package above, we can create a utility function BMICalculator which is a class in its own right that can calculate the weight of any person.

Now in the main program, we invoke this class to calculate the BMI of all the instances of the person created - p1 and p2 which are abstractions of the people John and Jane Doe.

package bmicalculator;

public class Program {

    public static void main(String[] args) {

      Person p1 = new Person("John", "Doe", 80, 1.7);
      Person p2 = new Person("Jane", "Doe", 65, 2.0);

    String p = Person.BMICalculator()
    }
}

The JVM compiles the code by parsing the public static key word and making it available to be called on all instances of the class and the objects created by that class as we see above.

What is dynamic binding, polymorphism and method overloading?

Polymorphism allows methods to behave differently depending on the context.

In Java, method overloading allows multiple methods to have the same name but different parameters.

In the Person class, we can demonstrate overloading where the signature of the method is different, the name can be “overloaded” with parameters as the method is called when the code is run and not when it is compiled. This is what is referred to as dynamic binding. Overloading is a form of “static” polymorphism as it is called within the class to run a method of the class rather than creating an instance of a class.

Dynamic polymorphism is when the class mutates and behaves like a completely different class when it creates new objects.

package bmicalculator;

class Person {

    private String name;
    private String surname;
    private Double weight;
    private Double height;

    // Constructor for initializing name and surname
    public Person(String name, String surname) {
        this.name = name;
        this.surname = surname;
    }

    // Constructor for initializing weight and height
    public Person(Double weight, Double height) {
        this.weight = weight;
        this.height = height;
    }

    // Method to calculate BMI based on the instance's weight and height
    public Double calcBMI() {
        if (height != null && weight != null && height > 0) {
            // Convert height to meters if it is in cm
            double heightInMeters = this.height / 100.0;
            return this.weight / (heightInMeters * heightInMeters);
        }
        return 0.0; // Return 0 if input is invalid
    }

    // Overloaded method to calculate BMI and print information about the person
    public void calcBMI(String name, String surname, Double weight, Double height) {
        System.out.println("This is the BMI of " + name + " " + surname);
        double heightInMeters = height / 100.0;  // Convert height to meters
        double bmi = weight / (heightInMeters * heightInMeters);
        System.out.println("Calculated BMI: " + bmi);
    }

    // Other methods of the class

}

    // Main method for testing
    public static void main(String[] args) {
        // Create a person with name and surname
        Person person1 = new Person("John", "Doe");
        
        // Create another person with weight and height
        Person person2 = new Person(70.0, 175.0); // weight in kg, height in cm

        // Test the BMI calculation using the instance method
        System.out.println(person2.getName() + " " + person2.getSurname() + "'s BMI: " + person2.calcBMI());

        // Test the overloaded BMI calculation method
        person2.calcBMI(person2.getName(), person2.getSurname(), person2.getWeight(), person2.getHeight());
    }
}

To elaborate, the first method in runtime will return the BMI of a person, while the second method will print the name of the person and their BMI. Both are methods of the class.

Because the method has the same name with different signatures, method overloading is an indication of the principle of polymorphism, adds Tusa(6), but should not be confused with dynamic polymorphism.

Derived from the Greek word meaning many forms - poly (many) and morph (form) - polymorphism in object-oriented programming refers to the ability of methods to adopt several forms, method overloading being one example.

What is inheritance in object-oriented programming?

Inheritance allows a subclass to inherit attributes and methods from a parent class. This promotes code reuse and organization. Inheritance can be used to extend classes and add specific behavior for different types of objects.

The principle of inheritance in object-oriented programming depends on the method hierarchy and in Java, the keyword super is used when a superclass is created. The sub-classes of this class will inherit the attributes and methods of the superclass. The classes “extend” the main class features.

In our example, if the hypothesis in real life that we want to check is whether young adults, mid-life adults and seniors change with age, we can extend the person class into 3 sub-classes

package bmicalculator;

public class YoungAdult extends Person {
    private int age;  // Instance variable for age

//initialise attributes
    public YoungAdult(String name, 
            String surname, 
            Double weight, 
            Double height, 
            int age) {
//`call super from parent
        super(name, surname, weight, height);
        this.age = age;  // Initialize age specific to the YoungAdult class
    }

    // Getter for age
    public int getAge() {
        return age;
    }

    // Other methods related to YoungAdult class can go here...
}

//  test main method
 
            // Create a YoungAdult object - add age 
        YoungAdult ya1 = new YoungAdult("Tom", "Jones", 59.2, 1.5, 18);
        // Access attributes via getter methods
        System.out.println("Name: " + ya1.getName() + " " + ya1.getSurname());
       System.out.println("BMI p2: " + ya1.getBMI());
        System.out.println("Age: " + ya1.getAge());
        

The second sub-class also inherits all the attributes of the parent class with the super key word. You can also add specificity, if you want to, for example, check if gender is a factor for BMI.

package bmicalculator;

public class Seniors extends Person {

    private int age;    // Additional attribute for Senior
    private String gender; // Additional attribute for Senior

    // Constructor for Senior class
    public Seniors(String name, 
            String surname, 
            Double weight, 
            Double height, 
            int age, 
            String gender) {
        // Call the parent constructor to initialize name, surname, weight, and height
        super(name, surname, weight, height); 
        this.age = age;         // Initialize age specific to Senior
        this.gender = gender;   // Initialize gender specific to Senior
    }

    // Getter method for gender
    public String getGender() {
        return gender;
    }

    // Getter method for age (optional, if needed) - or can be overriden
   public int getAge(int age) {
        if (age >60 ) {
     System.out.println("Caught an exception - age must be above 60");
        }
    return age;
    }

    // You can also add additional methods relevant to seniors here...
}

// check main method

       // Create a Senior object - add gender
        Seniors s1 = new Seniors("Janet", "Jones", 49.9, 1.2, "Female");

        // Access attributes via getter methods
        System.out.println("Name: " + s1.getName() + " " + s1.getSurname());
        System.out.println("BMI: " + s1.getBMI());
        System.out.println("Age: " + s1.getAge());
        System.out.println("Gender: " + s1.getGender());
        // as it is overrriden the value must be explicitly provided
       System.out.println("Gender: " + s1.getAge(55));

With the examples above, we have explored the core concepts that underly object-oriented programming - abstraction, encapsulation, static polymorphism and inheritance.

Demonstrating the advantages of object oriented programming

In the final code we can check the BMI of several categories of people in one instance. The main method now allows us to print the values of the BMI of all of these people.

package bmicalculator;

public class Program {

    public static void main(String[] args) {

        Person p1 = new Person("John", "Doe", 90.4, 1.7);
        System.out.println("BMI p1: " + p1.getBMI());
        
        Person p2 = new Person("Jane", "Doe", 55.0, 1.7);   
        System.out.println("BMI p2: " + p2.getBMI());
        
        YoungAdult ya1 = new YoungAdult("Tom", "Jones", 55.2, 1.5, 18);
        System.out.println("BMI ya1: " + ya1.getBMI());
        
        Seniors s1 = new Seniors("Janet", "Jones", 49.9, 1.5, 82, "Female");
        System.out.println("BMI s1: " + s1.getBMI());

    }

}

In the Person class, the code is modular and repeatable. This allows us now to change names and heights and weights and we will get an accurate BMI each time.

package bmicalculator;

public class Person {

    private String name;
    private String surname;
    private Double weight;  // weight in kilograms
    private Double height;  // height in meters
    private Double bmi;

    public Person(String name, String surname, Double weight, Double height) {
        this.name = name;
        this.surname = surname;
        this.weight = weight;
        this.height = height;
        this.bmi = calculateBMI();  // Calculate BMI when the object is created
    }

    private Double calculateBMI() {
        if (weight != null && height != null && height > 0) {
            return weight / (height * height);
        }
        return 0.0; // Return 0.0 if input values are invalid
    }

//methods getters - read data
    public String getName() {
        return name;
    }

    public String getSurname() {
        return surname;
    }

    //additional getter methods 
    public String getFullName() {
        return name + " " + surname;
    }

    public Double getBMI() {
        return bmi;
    }

    //mutate state - setter methods change params
    public String setName(String updatedName) {
        return updatedName;
    }

    public String setSurname(String updatedSurname) {
        return updatedSurname;
    }

    public String setFullName(String updatedName, String updatedSurname) {
        return updatedName + " " + updatedSurname;
    }

    public Double setWeight(Double updatedWeight) {
    return updatedWeight;
       
    }

    public Double setHeight(Double updatedHeight) {
     return updatedHeight;
    }

}

The code is extensible to new demographics of people we want to test BMIs for seniors and young adults. I have run the code on NetBeans to ensure the code executes as expected.

EXTERNAL REFERENCES