Continued from Unified Modelling Language
Encapsulation & Data Hiding in Object-Oriented Programming
Encapsulation is a mechanism in object-oriented programming (OOP) where both the data (variables) and the methods that operate on the data are bundled together as a single unit. This principle ensures that an object’s internal state is protected from unauthorized access or modification.
Key Concepts of Encapsulation:
- Private Variables: The class variables (also known as attributes or fields) are declared as
private
. This means they cannot be accessed directly from outside the class. - Getter and Setter Methods: Access to these private variables is provided through getter and setter methods. These methods allow controlled access to the private data.
Getter and Setter Methods
- Getter Method: Used to retrieve the value of a private variable.
- Setter Method: Used to update the value of a private variable.
Naming Conventions:
- The getter and setter methods are generally named by prefixing the variable name with
get
andset
, respectively.
Encapsulation is also known as data hiding because it hides the internal details of how the data is stored or modified, exposing only the necessary interface for interacting with the object.
Example: Implementing Encapsulation in Java
// Define the Person class
public class Person {
// Declare private variable 'age'. This will not be directly accessible from outside the class
private int age;
// Getter method for the 'age' variable
// This method allows external access to the value of 'age' in a controlled manner
public int getAge() {
return age; // Return the current value of 'age'
}
// Setter method for the 'age' variable
// This method allows external code to modify the value of 'age' but with validation
public void setAge(int age) {
// Validation: Ensure the age is within a valid range (between 0 and 100)
if (age >= 0 && age <= 100) {
this.age = age; // If valid, set the value of 'age'
} else {
// If the age is not valid, print an error message
System.out.println("Invalid age! Age must be between 0 and 100.");
}
}
// Main method to demonstrate the use of encapsulation
public static void main(String[] args) {
// Create an object of the Person class
Person person = new Person();
// Set the 'age' value using the setter method
person.setAge(25); // This is a valid age
System.out.println("Age: " + person.getAge()); // Get the 'age' value using the getter method
// Attempt to set an invalid age
person.setAge(150); // This is an invalid age
System.out.println("Age: " + person.getAge()); // The value of 'age' should remain unchanged
}
}
Explanation:
- The
Person
class contains a private variableage
, ensuring that it is protected from direct access from outside the class. - The
getAge()
method is a getter method that allows other classes or code to retrieve the value ofage
in a controlled manner. - The
setAge()
method is a setter method that allows modification of theage
variable but includes validation to ensure that the age is within a valid range (between 0 and 100). If an invalid age is provided, an error message is printed, and theage
remains unchanged. - In the
main()
method, we create an instance of thePerson
class and demonstrate how the getter and setter methods work.
Output:
Age: 25
Invalid age! Age must be between 0 and 100.
Age: 25
Nested Classes in Java
In Java, when a class is defined inside another class, it is known as a nested class. A nested class can be either static or non-static (inner class).
Static Nested Class
A static nested class is defined with the static
keyword. Static nested classes can be instantiated without needing an instance of the outer class.
// Define the Animal class
class Animal {
// Static nested class Mammal
static class Mammal {
// Method to display information about mammals
public void displayInfo() {
System.out.println("I am a mammal");
}
}
}
// Main method to demonstrate the usage of static nested class
public class Main {
public static void main(String[] args) {
// Create an instance of the static nested class Mammal without an instance of Animal
Animal.Mammal mammal = new Animal.Mammal();
mammal.displayInfo(); // Calling the method to display info
}
}
Explanation:
- The
Mammal
class is a static nested class inside theAnimal
class. Since it is static, we can instantiate it without needing an instance ofAnimal
. - In the
Main
class, we create an instance ofMammal
using the syntaxAnimal.Mammal
and call thedisplayInfo()
method.
Non-Static Nested Class (Inner Class)
A non-static nested class, also known as an inner class, is defined inside an outer class and has access to the outer class’s variables and methods, even if they are private.
// Outer class
class Outer {
// Inner class defined inside the Outer class
class Inner {
// Method inside the Inner class
public void show() {
System.out.println("In a nested class method");
}
}
}
// Main method to demonstrate the usage of inner class
public class MethodDemo {
public static void main(String[] args) {
// Create an instance of the Outer class
Outer outer = new Outer();
// Create an instance of the inner class by first creating the outer class instance
Outer.Inner inner = outer.new Inner();
// Call the method inside the Inner class
inner.show();
}
}
Explanation:
- The
Inner
class is defined inside theOuter
class. To create an instance of theInner
class, we must first create an instance of theOuter
class, as the inner class depends on the outer class’s instance. - In the
MethodDemo
class, we first instantiate theOuter
class, then use that instance to create theInner
class instance. Theshow()
method of theInner
class is called to display a message.
Example 1: Nested Static and Non-Static Classes
class Animal {
// Static nested class Mammal
static class Mammal {
// Method to display info for Mammal
public void displayInfo() {
System.out.println("I am a mammal.");
}
}
// Non-static nested class Reptile
class Reptile {
// Method to display info for Reptile
public void displayInfo() {
System.out.println("I am a reptile.");
}
// Method to eat for Reptile
public void eat() {
System.out.println("I eat food.");
}
}
// Method to eat for Animal
public void eat() {
System.out.println("I eat food.");
}
}
public class Main {
public static void main(String[] args) {
// Create an instance of Animal
Animal animal = new Animal();
// Create an instance of Reptile using the Animal instance
Animal.Reptile reptile = animal.new Reptile();
reptile.displayInfo();
reptile.eat(); // Correct usage
// Creating an instance of static nested class Mammal
Animal.Mammal mammal = new Animal.Mammal();
mammal.displayInfo();
// Calling the eat() method from Animal
animal.eat(); // Correct usage
}
}
Explanation:
- The
Animal
class has both a static nested classMammal
and a non-static nested classReptile
. - The
Mammal
class is static, so it can be instantiated without anAnimal
object. - The
Reptile
class is non-static, so it requires anAnimal
instance to create aReptile
object. - Both nested classes and the main class demonstrate the use of the
displayInfo()
andeat()
methods.
Example 2: Local Inner Classes (Non-Static)
class Outer {
// Method in the Outer class
void outerMethod() {
System.out.println("inside Outer class");
// Inner class defined inside the outerMethod (Local inner class)
class Inner {
// Method in the Inner class
void innerMethod() {
System.out.println("inside Inner class");
}
}
// Create an instance of Inner class
Inner y = new Inner();
y.innerMethod(); // Calls the method from Inner class
}
}
public class MethodDemo {
public static void main(String[] args) {
// Create an instance of Outer class
Outer x = new Outer();
x.outerMethod(); // Calls the outer method
// The following line would give an error because Inner class is local to outerMethod
// Outer.Inner in = Outer.new Inner(); // Error: Cannot reference Inner from outside outerMethod
}
}
Explanation:
- The
Inner
class is defined within theouterMethod
of theOuter
class. This makes it a local inner class, which can only be accessed within the method where it is defined. - The
MethodDemo
class creates an instance ofOuter
and calls theouterMethod()
. InsideouterMethod()
, theInner
class is instantiated and itsinnerMethod()
is called.
Output for both examples:
Example 1:
I am a reptile.
I eat food.
I am a mammal.
I eat food.
Example 2:
inside Outer class
inside Inner class
Advantages of Inner Classes in the Code:
// Outer class that encapsulates data and functionality
class Outer {
private int value; // private member variable
// Constructor to initialize the value
public Outer(int value) {
this.value = value;
}
// Getter method for 'value'
public int getValue() {
return value;
}
// Inner class that provides additional functionality
class Inner {
// Method in the inner class that accesses private members of the outer class
public void show() {
// Accessing the private member 'value' from the outer class
System.out.println("The value is: " + value);
}
}
// Method to demonstrate the encapsulation and organization
public void demonstrateInnerClass() {
// Creating an instance of the inner class and calling its method
Inner inner = new Inner();
inner.show(); // Calls the 'show' method of the inner class
}
}
public class Main {
public static void main(String[] args) {
// Create an instance of the Outer class
Outer outer = new Outer(10);
// Access the outer class functionality and inner class functionality
outer.demonstrateInnerClass();
}
}
-
Security with Private Data
- The
Inner
class has direct access to the private variables of the outer class (value
in this case), making it possible to encapsulate sensitive data securely and expose it only to the inner class when needed.
- The
-
Packaging of the Class:
- The
Inner
class is closely tied to its outer class (Outer
). This helps package related functionality together, keeping the logic of the application encapsulated within a specific outer class.
- The
-
Encapsulation:
- Inner classes help in encapsulating the functionality related to the outer class. The
Inner
class provides additional functionality while still being able to access the private members of theOuter
class.
- Inner classes help in encapsulating the functionality related to the outer class. The
-
Readable and Maintainable Code:
- By using inner classes, related functionality can be organized in a way that makes the code more readable and easier to maintain. The
Inner
class is used to encapsulate the code that is specific to the outer class but still logically part of the same context.
- By using inner classes, related functionality can be organized in a way that makes the code more readable and easier to maintain. The
-
Organize Package Structure:
- Inner classes help in keeping the related classes organized within the same file or package, which reduces the complexity and improves the package structure, especially when the inner class is tightly coupled with the outer class.
Output:
The value is: 10
Conclusion
Encapsulation and data hiding are powerful concepts that help secure and manage the internal state of objects. By using getter and setter methods, you can ensure that data is accessed and modified safely. Nested and inner classes allow you to organize code effectively, with static and non-static classes offering different levels of flexibility and access.
References
- Continued to Inheritance
Information
- date: 2025.02.24
- time: 08:12