Object Oriented Programming in Java
Object Oriented Programming in Java
in Java
Object Oriented Programming (OOP):-
Object Oriented Programming v/s Procedural Programming
Difference Between Java and C++ in Terms of Language Functions
Java Virtual Machine (JVM)
Key Roles of the JVM
Compilation and Interpretation in Java
Advantages of JVM and Compilation-Interpretation Model
Object v/s Object Reference:-
Main Characteristics of an OOP Language
1. Encapsulation
2. Abstraction
3. Inheritance
4. Polymorphism
Summary Table:-
Data Encapsulation v/s Data Abstraction
Association, Aggregation and Composition in OOP:-
Association
Aggregation
Composition
Differences between Association, Aggregation, and Composition
Link and Association
Basic Concepts in OOP
public static void main (String[] args)
1. public
2. static
3. void
4. main
5. String[] args
Example of main Method
Parameter Passing in Programming
Call by Value:-
Call by Reference:-
static Keyword in Java
Static Variables
Static Methods
Encapsulation: Bundling data and methods that operate on the data within
a single unit, or object, to protect data from outside interference.
Examples: Common OOP languages include Java, C++, Python, and C#.
Focuses on functions/
Focuses on objects and classes,
Program procedures, organizing code
organizing code around real-
Structure around tasks or steps to be
world entities
completed
Point of
Java C++
Differentiation
Platform-dependent; code is
Platform-independent, meaning
compiled into machine code
code is compiled into bytecode
Platform specific to the operating system,
that runs on the Java Virtual
Independence so executables need
Machine (JVM), making it cross-
recompilation for different
platform
platforms
1. Java Compiler: The java compiler ( javac ) converts Java source code (in
.java files) into bytecode (in .class files).
Example:
1. Class Loader: When the JVM executes the program, it first loads the .class
Example:
2. Object Reference:
It does not hold the actual data, just a reference (or pointer) to the
object.
Example:
1. Encapsulation
Definition: Encapsulation is the technique of bundling data (attributes) and
methods (functions) that operate on the data into a single unit, known as a
class. It restricts direct access to some components, which can protect the
integrity of the data.
Example: Using private variables and providing public getter and setter
methods to access and update them.
3. Inheritance
Definition: Inheritance is a mechanism that allows one class to inherit the
fields and methods of another class. The class that inherits is called the
subclass (or child class), and the class being inherited from is called the
super class (or parent class).
Example: A Dog class can inherit from an Animal class, meaning that Dog
class Animal {
void eat() {
System.out.println("Animal eats");
}
4. Polymorphism
Definition: Polymorphism allows a single method, action, or operator to
behave differently based on the object or context. In OOP, it refers to the
ability of different classes to respond to the same method call in their own
way.
Types:
Example:
class Animal {
void sound() {
System.out.println("Animal sound");
}
Summary Table:-
OOP
Description Purpose
Characteristic
How It’s Using private fields with public Using abstract classes, interfaces,
Achieved getters and setters and abstract methods
Characteristics:
Aggregation
Definition: Aggregation is a special form of association where one class
(the “whole”) contains a reference to another class (the “part”), but both
can exist independently.
Characteristics:
Composition
Definition: Composition is a stronger form of aggregation where one class
is entirely dependent on another for existence, indicating ownership.
Characteristics:
The “part” cannot exist without the “whole”; if the “whole” is destroyed,
the “part” is also destroyed.
Dependent; part is
Objects are fully Objects can exist
Independence destroyed with the
independent separately
whole
Strength of
Weak Moderate Strong
Bond
Teacher ↔
Example Library ↔ Book House ↔ Room
Student
In summary:
Link:
For example, if there is a Student object and a Course object, a link exists
if a specific student is enrolled in a specific course.
Association:
1. public
Access Modifier: public means that the main method is accessible from
outside the class, including by the Java Runtime Environment (JRE).
2. static
Static Keyword: static means that the main method belongs to the class
itself, rather than any specific instance of the class.
Purpose: This allows the JRE to call the main method without needing to
create an instance of the class. Without static , the JRE would have to
instantiate the class before it could call main , which would complicate the
program execution.
3. void
Return Type: void means that main does not return any value.
Purpose: Since the main method is only meant to start the program, there is
no need for it to return any data.
4. main
Method Name: main is a predefined name that the Java compiler and
runtime recognize as the starting point for program execution.
Purpose: The Java specification requires that this method be named main
5. String[] args
Parameter: String[] args is an array of String objects, allowing the program
to accept command line arguments.
"world"] .
if (args.length > 0) {
System.out.println("Arguments provided:");
for (String arg : args) {
System.out.println(arg);
}
} else {
System.out.println("No arguments provided");
}
}
}
>java Main
Hello, World
No arguments provided
2. Call by Reference: A reference to the actual parameter (not the value itself)
is passed, meaning that modifications in the function will affect the original
parameter.
Call by Value:-
The function receives a copy of the variable’s value.
In Java, primitive types (like int , float , char ) are passed by value.
In this example, the value of num in the main function remains unchanged after
calling the increment method.
Output:
Explanation: num remains 10 in the main method, because only a copy of its
value was passed to the increment function.
Call by Reference:-
The function receives a reference (or address) to the variable.
class Box {
int height;
Box (int h) {
this.height = h;
}
}
Output:
Static Variables
Declared with the static keyword within a class, static variables are shared
across all instances of that class.
class Example {
static int count = 0; // Shared across all instances
}
Static Methods
class MathUtility {
static int add (int a, int b) {
return a + b;
}
}
class Example {
public static void main (String args[]) {
int r = MathUtility.add(4,5); // Calling the method u
System.out.println(r); // Output: 9
}
}
Static Blocks
A static block is a block of code that gets executed when the class is
loaded, before any objects are created.
class Example {
static int count;
static {
count = 10; // Initialization in a static block
System.out.println("Static block executed");
}
}
class Outer {
static class StaticNested {
void display() {
System.out.println("Inside static nested class");
}
}
}
class Example {
public static void main (String args[]) {
Outer.StaticNested nestedInstance = new Outer.StaticNe
nestedInstance.display(); // Output: Inside static ne
}
}
Access: Static members can be accessed directly using the class name,
without needing an object instance.
Static Fields: A static field is shared across all instances of the class.
Changing it in one instance affects all others.
Example:
class Example {
static int count = 0; // Shared among all instances
static void showCount() {
System.out.println(count);
}
}
2. final :
Example:
class Example {
final int MAX_COUNT = 100; // Cannot be modified
final void display() {
System.out.println("This method cannot be overri
Difference:
staticis associated with the class itself, meaning it is shared across all
instances.
3. Loose Coupling:
class Engine {
void start() {
System.out.println("Engine starts");
}
}
class Car {
private Engine engine = new Engine(); // Aggregation (Car
void startCar() {
engine.start(); // Car sends a message to Engine to s
}
}
In this example:
The Car object doesn’t access the Engine 's details directly but
communicates by calling engine.start() .
This start() method call is the message sent to the Engine object.
Method Overloading
Method Overloading is a feature in Java that allows a class to have multiple
methods with the same name but different parameter lists (different types,
number of parameters, or order of parameters). This allows methods to
perform similar functions with varied input.
Benefits: It improves the clarity of the code and enables better reusability
by allowing related operations to be expressed using the same method
once.
Example:
class MathOperations {
int add (int a, int b) {
return a + b;
}
In this example, the add method is overloaded to handle both integer and
double types, as well as different number of parameters.
Benefits: Garbage collection reduces the risk of memory leaks and ensures
efficient memory utilization, which is crucial in long-running applications.
How it Works When the garbage collector determines that there are no
more references to an object, it calls the finalize() method once on that
object (if it exists) before reclaiming the memory. This gives the object a
last opportunity to release resources or perform other cleanup tasks.
Limitations: Since the garbage collector does not guarantee when it will
run, relying on finalize() method for critical cleanup tasks is discouraged.
Starting from Java 9, it is considered deprecated due to its unpredictability,
and alternatives like try-with-resources and finally blocks are recommended
instead.
Example:
class MyClass {
@Override
protected void finalize() {
System.out.println("Cleaning up resources...");
}
}
Inheritance in Java
Types of Inheritance in OOP
In object-oriented programming, inheritance is the mechanism that allows one
class to inherit properties and behaviors (methods and fields) from another
class. This facilitates code reuse, improves organization, and supports
polymorphism. The various types of inheritance are as follows:-
Single Inheritance
Definition: A class inherits from only one superclass. This means each
class has one direct parent.
Example:
class Animal {
void eat() {
System.out.println("Eating...");
}
}
Here, Dog inherits from Animal , but Animal has no other parent classes.
Issues: This can lead to the diamond problem, where ambiguity arises if
multiple parents have methods with the same name.
Multilevel Inheritance
Definition: A class inherits from a superclass, which in turn inherits from
another superclass. This creates a chain of inheritance.
Example:
class Animal {
void eat() { System.out.println("Eating..."); }
}
Here, Puppy inherits from Dog , and Dog inherits from Animal , creating a
multilevel inheritance chain.
Hierarchical Inheritance
Definition: Multiple classes inherit from a single superclass, making each
subclass independent of the others but sharing a common parent.
Example:
class Animal {
void eat() { System.out.println("Eating..."); }
}
Here, both Dog and Cat inherit from Animal , making Animal a common parent.
Hybrid Inheritance
Definition: A combination of two or more types of inheritance (e.g.,
hierarchical and multilevel).
class Animal {
void sound() {
System.out.println("Animal sound");
}
}
The super() call must be the first line in the subclass constructor if used.
class Animal {
String type = "Mammal";
}
void printType() {
System.out.println(super.type); // Accesses Animal's
}
}
The method in the subclass must have the same name, return type, and
parameter list as in the superclass.
Example:
class Vehicle {
void start () {
System.out.println("Vehicle starts");
}
}
Explanation: The Car class overrides the start method of Vehicle , and
when called, it executes the Car class’s implementation due to dynamic
method dispatch.
Cannot declare any new checked exceptions that were not declared in
the superclass method.
If the superclass method does not declare any checked exceptions, then
the overriding method in the subclass also cannot declare any checked
exceptions.
Example:-
Suppose we have a superclass method that doesn’t declare any exceptions:
class SuperClass {
void display() {
System.out.println("Superclass display method");
class Subclass {
@Override
void display() throws ArithmeticException { // Allowed si
System.out.println("Subclass display method");
throw new ArithmeticException("Unchecked Exception");
}
}
Summary:-
With Covariant Return Type: The subclass can return a type derived from
the superclass’s return type, as long as it satisfies the requirements of the
superclass’s return type.
Example:
class Animal {
Animal get() {
return this;
}
}
Explanation: Here, Dog 's overridden get() method returns Dog , a subclass
of Animal , which is permitted due to covariant return types.
1. public
Description: The public modifier allows a class, method, or variable to be
accessible from any other class in any package.
Usage:
Example:
2. protected
Description: The protected modifier allows visibility within the same
package and in subclasses (even if they are in different packages).
Usage: Useful when you want to give access to subclasses but prevent
access from unrelated classes outside the package.
Example:
3. private
Description: The private modifier restricts visibility to within the same class
only. It prevents access from any other class, including subclasses and
classes in the same package.
Usage: Commonly used for encapsulating data, like instance variables and
helper methods that should not be accessible from outside the class.
Example:
Example:
class MyClass {
int value = 40; // Default access, accessible within the
void display() {
System.out.println("Default Method");
}
}
Summary Table:-
World (Other
Modifier Class Package Subclass
Packages)
public ✅ ✅ ✅ ✅
protected ✅ ✅ ✅ ❌
default ✅ ✅ ❌ ❌
private ✅ ❌ ❌ ❌
How it Works: Java achieves this by looking up the method in the subclass
if a method is overridden. This process enables polymorphic behavior.
Example:
class Animal {
void sound () {
System.out.println("Animal sound");
}
}
Example:-
Early Binding:
class Example {
static void display() { // Static method - early bin
System.out.println("Static display method");
}
}
class EarlyBinding {
Late Binding:
class Animal {
void sound() { // Instance method - eligible for lat
System.out.println("Animal sound");
}
}
class LateBinding {
public static void main (String args[]) {
Animal animal = new Dog();
animal.sound(); // resolved at runtime based on
}
}
Abstraction:-
Definition: Abstraction is one of the core principles of Object-Oriented
Programming (OOP), focusing on hiding unnecessary details while
exposing only essential features relevant to the user. It allows developers to
manage complexity by providing a simplified model of the system.
How it Works:
Advantages of Abstraction:
Enhanced Code Reusability: Abstraction allows for code reuse since
common functionality can be defined in an abstract class or interface.
Levels of Abstraction:
Low-Level Abstraction: Focuses on the internal, detailed working of the
system. Example: bits and bytes in memory management.
Example Code:
abstract class Vehicle {
abstract void move();
}
Applications:
Used in software design to build modular, scalable, and maintainable
applications.
Can have instance fields of any All fields are public , static ,
Field Type
access level and final by default (constants)
Example:
Abstract Class:
Interface
interface Flyable {
void fly(); // Abstract method by default
}
interface Printable {
void print(); // Abstract method
}
interface Showable {
@Override
public void show() {
System.out.println("Showing document...");
}
}
Explanation:-
2. Class Implementation:
Packages in Java
In Java, a package is a namespace that groups related classes, interfaces, and
sub-packages. Packages help organize code, prevent naming conflicts, and
control access to classes and methods, improving modularity and reusability.
There are two types of packages in java:
package com.example.utilities;
Importing a Package: Use the import keyword to access classes from other
packages.
import com.example.utilities.Utility;
1. try : A block of code that might throw an exception is placed inside the try
2. catch : Used to handle the exception. Each catch block can catch a specific
type of exception. A try block can have multiple catch blocks to handle
different exceptions.
ArithmeticException("Division by zero"); .
Output:
Types of Exceptions
Checked Exceptions
These are exceptions that the compiler checks during compilation, requiring
them to be either handled with try-catch or declared in the method signature
with throws .
Example:
Example:
Errors
Errors are serious issues beyond the application’s control. They are not meant
to be caught or handled by programs.
Custom Exceptions
Java allows developers to create their own custom exceptions by extending the
Exception class. This is useful when the application has unique conditions that
aren’t covered by built-in exceptions.
Output:
2. Use Finally to Clean Up Resources: Always close resources like files and
database connections in the finally block or use try-with-resources .
Syntax Example
Output:
Threading in Java
Memory Each process has a separate Threads share the memory and
Allocation memory space. resources of the parent process.
Creation Time Slower to create and initialize. Faster to create and initialize.
Explanation:
The MyThread class extends Thread and overrides the run() method, which
specifies the code to be executed by the thread.
The start() method begins the execution of the thread and calls run()
internally.
Explanation:
2. Active: After calling start() , the thread enters the active state. Contains
two states within it:
b. Running: When the thread gets the CPU, it moves from the runnable to
the running state.
3. Blocked/ Waiting: A thread enters this state when waiting for resources or
another thread.
5. yield() : Pauses the currently executing thread to allow other threads to run.
Thread Synchronization
Synchronization is essential when multiple threads access shared resources,
to prevent data inconsistency. Java provides synchronized methods and blocks
to achieve this.
Example:
class Counter {
private int count = 0;
t1.start();
t2.start();
t1.join();
t2.join();
Explanation:
increment() is synchronized, ensuring that only one thread can modify count
at a time.
t1.join()and t2.join() ensure that the main thread waits until both threads
complete execution.
class SharedResource {
private int data;
private boolean available = false;
producer.start();
consumer.start();
}
}
Explanation:
Producer generates data and calls produce() , which stores data and notifies
the consumer.
Consumer waits for data using consume() , retrieves it, and notifies the
producer.
Deadlock
A deadlock occurs when two or more threads are waiting for each other to
release resources, causing them to be stuck indefinitely.
Example:
t1.start();
t2.start();
}
}
In this example, t1 and t2 each acquire a lock and wait for the other, causing
a deadlock.