The Main Method

Download as docx, pdf, or txt
Download as docx, pdf, or txt
You are on page 1of 41

The main Method

To run our program, the main method must be identical to this


signature:public static void main(String[ ] args)
- public: anyone can access it
- static: method can be run without creating an instance of the class
containing the main method
- void: method doesn't return any value
- main: the name of the method

For example, the following code declares a method called test, which does
not return anything and has no parameters:void test()
The method's parameters are declared inside the parentheses that follow
the name of the method.
For main, it's an array of strings called args. We will use it in our next
lesson, so don't worry if you don't understand it all now.

String Concatenation

The + (plus) operator between strings adds them together to make a new
string. This process is called concatenation.
The resulted string is the first string put together with the second string.
For example:
String firstName, lastName;
firstName = "David";
lastName = "Williams";

System.out.println("My name is " + firstName +" "+lastName);

// Prints: My name is David WilliamsTry It Yourself

The char data type represents a single character.

Getting User Input

While Java provides many different methods for getting user input,
the Scanner object is the most common, and perhaps the easiest to
implement. Import the Scanner class to use the Scanner object, as seen
here:import java.util.Scanner;
In order to use the Scanner class, create an instance of the class by using
the following syntax:Scanner myVar = new Scanner(System.in);
You can now read in different kinds of input data that the user enters.
Here are some methods that are available through the Scanner class:
Read a byte - nextByte()
Read a short - nextShort()
Read an int - nextInt()
Read a long - nextLong()
Read a float - nextFloat()
Read a double - nextDouble()
Read a boolean - nextBoolean()
Read a complete line - nextLine()
Read a word - next()

Example of a program used to get user input:


import java.util.Scanner;

class MyClass {
public static void main(String[ ] args) {
Scanner myVar = new Scanner(System.in);
System.out.println(myVar.nextLine());
}
}Try It Yourself

This will wait for the user to input something and print that input.
The code might seem complex, but you will understand it all in the
upcoming lessons.

The switch Statement

A switch statement tests a variable for equality against a list of values.


Each value is called a case, and the variable being switched on is checked
for each case.
Syntax:switch (expression) {
case value1 :
//Statements
break; //optional
case value2 :
//Statements
break; //optional
//You can have any number of case statements.
default : //Optional
//Statements
}
- When the variable being switched on is equal to a case, the statements
following that case will execute until a break statement is reached.
- When a break statement is reached, the switch terminates, and the
flow of control jumps to the next line after the switch statement.
- Not every case needs to contain a break. If no break appears, the flow
of control will fall through to subsequent cases until a break is reached.

The example below tests day against a set of values and prints a
corresponding message.
int day = 3;

switch(day) {
case 1:
System.out.println("Monday");
break;
case 2:
System.out.println("Tuesday");
break;
case 3:
System.out.println("Wednesday");
break;
}
// Outputs "Wednesday"Try It Yourself

You can have any number of case statements within a switch.


Each case is followed by the comparison value and a colon.

for Loops

Another loop structure is the for loop. A for loop allows you to efficiently
write a loop that needs to execute a specific number of times.
Syntax:for (initialization; condition; increment/decrement) {
statement(s)
}
Initialization: Expression executes only once during the beginning of loop
Condition: Is evaluated each time the loop iterates. The loop executes the
statement repeatedly, until this condition returns false.
Increment/Decrement: Executes after each iteration of the loop.

The following example prints the numbers 1 through 5.


for(int x = 1; x <=5; x++) {
System.out.println(x);
}

/* Outputs
1
2
3
4
5
*/Try It Yourself

This initializes x to the value 1, and repeatedly prints the value of x, until
the condition x<=5 becomes false. On each iteration, the statement x++
is executed, incrementing x by one.
Notice the semicolon (;) after initialization and condition in the syntax.

Loop Control Statements

The break and continue statements change the loop's execution flow.
The break statement terminates the loop and transfers execution to the
statement immediately following the loop.
Example:
int x = 1;

while(x > 0) {
System.out.println(x);
if(x == 4) {
break;
}
x++;
}

/* Outputs
1
2
3
4
*/Try It Yourself

The continue statement causes the loop to skip the remainder of its body
and then immediately retest its condition prior to reiterating. In other
words, it makes the loop skip to its next iteration.
Example:
for(int x=10; x<=40; x=x+10) {
if(x == 30) {
continue;
}
System.out.println(x);
}
/* Outputs
10
20
40
*/Try It Yourself

As you can see, the above code skips the value of 30, as directed by
the continue statement.

Arrays

An array is a collection of variables of the same type.


When you need to store a list of values, such as numbers, you can store
them in an array, instead of declaring separate variables for each number.

To declare an array, you need to define the type of the elements


with square brackets.
For example, to declare an array of integers:int[ ] arr;
The name of the array is arr. The type of elements it will hold is int.

Now, you need to define the array's capacity, or the number of elements it
will hold. To accomplish this, use the keyword new.int[ ] arr = new int[5];
The code above declares an array of 5 integers.
In an array, the elements are ordered and each has a specific and constant
position, which is called an index.

To reference elements in an array, type the name of the array followed by


the index position within a pair of square brackets.
Example:arr[2] = 42;
This assigns a value of 42 to the element with 2 as its index.
Note that elements in the array are identified with zero-based index
numbers, meaning that the first element's index is 0 rather than one. So,
the maximum index of the arrayint[5] is 4.

Initializing Arrays

Java provides a shortcut for instantiating arrays of primitive types and


strings.
If you already know what values to insert into the array, you can use
an array literal.
Example of an array literal:
String[ ] myNames = { "A", "B", "C", "D"};
System.out.println(myNames[2]);

// Outputs "C"Try It Yourself

Place the values in a comma-separated list, enclosed in curly braces.


The code above automatically initializes an array containing 4 elements,
and stores the provided values.
Sometimes you might see the square brackets placed after
the array name, which also works, but the preferred way is to place the
brackets after the array's data type.

Enhanced for Loop

The enhanced for loop (sometimes called a "for each" loop) is used to
traverse elements in arrays.
The advantages are that it eliminates the possibility of bugs and makes
the code easier to read.
Example:
int[ ] primes = {2, 3, 5, 7};

for (int t: primes) {


System.out.println(t);
}

/*
2
3
5
7
*/Try It Yourself

The enhanced for loop declares a variable of a type compatible with the
elements of the arraybeing accessed. The variable will be available within
the for block, and its value will be the same as the current array element.
So, on each iteration of the loop, the variable t will be equal to the
corresponding element in the array.
Notice the colon after the variable in the syntax.

Multidimensional Arrays
Multidimensional arrays are array that contain other arrays. The two-
dimensional array is the most basic multidimensional array.
To create multidimensional arrays, place each array within its own set of
square brackets. Example of a two-dimensional array:int[ ][ ] sample =
{ {1, 2, 3}, {4, 5, 6} };
This declares an array with two arrays as its elements.
To access an element in the two-dimensional array, provide two indexes,
one for the array, and another for the element inside that array.
The following example accesses the first element in the second array of
sample.
int x = sample[1][0];
System.out.println(x);

// Outputs 4Try It Yourself

The array's two indexes are called row index and column index.

Multidimensional Arrays

You can get and set a multidimensional array's elements using the same
pair of square brackets.
Example:
int[ ][ ] myArr = { {1, 2, 3}, {4}, {5, 6, 7} };
myArr[0][2] = 42;
int x = myArr[1][0]; // 4Try It Yourself

The above two-dimensional array contains three arrays. The first array has
three elements, the second has a single element and the last of these has
three elements.
In Java, you're not limited to just two-dimensional arrays. Arrays can be
nested within arrays to as many levels as your program needs. All you
need to declare an array with more than two dimensions, is to add as
many sets of empty brackets as you need. However, these are harder to
maintain.
Remember, that all array members must be of the same type.

Object-Orientation

Java uses Object-Oriented Programming (OOP), a programming style that is


intended to make thinking about programming closer to thinking about the real
world.
In OOP, each object is an independent unit with a unique identity, just as
objects in the real world are.
An apple is an object; so is a mug. Each has its unique identity. It's possible to
have two mugs that look identical, but they are still separate, unique objects.
Objects also have characteristics, which are used to describe them.
For example, a car can be red or blue, a mug can be full or empty, and so on.
These characteristics are also called attributes. An attribute describes the
current state of an object.
In the real world, each object behaves in its own way. The car moves, the phone
rings, and so on.
The same applies to objects: behavior is specific to the object's type.
In summary, in object oriented programming, each object has three
dimensions: identity, attributes, and behavior.
Attributes describe the object's current state, and what the object is capable of
doing is demonstrated through the object's behavior.

Classes

A class describes what the object will be, but is separate from the object itself.
In other words, classes can be described as blueprints, descriptions, or definitions
for an object. You can use the same class as a blueprint for creating multiple
objects. The first step is to define the class, which then becomes a blueprint for
object creation.

Each class has a name, and each is used to define attributes and behavior.
Some examples of attributes and behavior:
Methods

Methods define behavior. A method is a collection of statements that are


grouped together to perform an operation. System.out.println() is an
example of a method.
You can define your own methods to perform your desired tasks.
Let's consider the following code:
class MyClass {

static void sayHello() {


System.out.println("Hello World!");
}

public static void main(String[ ] args) {


sayHello();
}
}
// Outputs "Hello World!"

Method Parameters

You can also create a method that takes some data, called parameters,
along with it when you call it. Write parameters within the method's
parentheses.
For example, we can modify our sayHello() method to take and output
a String parameter.
class MyClass {

static void sayHello(String name) {


System.out.println("Hello " + name);
}

public static void main(String[ ] args) {


sayHello("David");
sayHello("Amy");
}

}
// Hello David
// Hello AmyTry It Yourself

The method above takes a String called name as a parameter, which is


used in the method's body. Then, when calling the method, we pass the
parameter's value inside the parentheses.
Methods can take multiple, comma-separated parameters.
The advantages of using methods instead of simple statements include
the following:
- code reuse: You can write a method once, and use it multiple times,
without having to rewrite the code each time.
- parameters: Based on the parameters passed in, methods can perform
various actions.

public static void main(String[ ] args)


This definition indicates that the main method takes an array of Strings as its
parameters, and does not return a value.

Access Modifiers

Now let's discuss the public keyword in front of the main method.public
static void main(String[ ] args)
public is an access modifier, meaning that it is used to set the level of
access. You can use access modifiers for classes, attributes, and methods.

For classes, the available modifiers are public or default (left blank), as
described below:
public: The class is accessible by any other class.
default: The class is accessible only by classes in the same package.

The following choices are available for attributes and methods:


default: A variable or method declared with no access control modifier is
available to any other class in the same package.
public: Accessible from any other class.
protected: Provides the same access as the default access modifier, with
the addition that subclasses can access protected methods and variables
of the superclass (Subclasses and superclasses are covered in upcoming
lessons).
private: Accessible only within the declared class itself.

Example:public class Vehicle {


private int maxSpeed;
private int wheels;
private String color;
private double fuelCapacity;

public void horn() {


System.out.println("Beep!");
}
}
It's a best practice to keep the variables within a class private. The
variables are accessible and modified using Getters and Setters.
Tap Continue to learn about Getters and Setters.

Getters & Setters

Getters and Setters are used to effectively protect your data, particularly
when creating classes. For each variable, the get method returns its value,
while the set method sets the value.

Getters start with get, followed by the variable name, with the first letter
of the variable name capitalized.
Setters start with set, followed by the variable name, with the first letter
of the variable name capitalized.

Example:public class Vehicle {


private String color;

// Getter
public String getColor() {
return color;
}

// Setter
public void setColor(String c) {
this.color = c;
}
}
The getter method returns the value of the attribute.
The setter method takes a parameter and assigns it to the attribute.
The keyword this is used to refer to the current object.
Basically, this.color is the color attribute of the current object.
Getters & Setters

Once our getter and setter have been defined, we can use it in our main:
public static void main(String[ ] args) {
Vehicle v1 = new Vehicle();
v1.setColor("Red");
System.out.println(v1.getColor());
}

//Outputs "Red"Try It Yourself

Getters and setters allow us to have control over the values. You may, for
example, validate the given value in the setter before actually setting the
value.
Getters and setters are fundamental building blocks for encapsulation,
which will be covered in the next module.

Constructors

Constructors are special methods invoked when an object is created and


are used to initialize them.
A constructor can be used to provide initial values for object attributes.

- A constructor name must be same as its class name.


- A constructor must have no explicit return type.

Example of a constructor:public class Vehicle {


private String color;
Vehicle() {
color = "Red";
}
}
The Vehicle() method is the constructor of our class, so whenever an
object of that class is created, the color attribute will be set to "Red".
A constructor can also take parameters to initialize attributes.public class
Vehicle {
private String color;
Vehicle(String c) {
color = c;
}
}

Using Constructors

The constructor is called when you create an object using


the new keyword.
Example:public class MyClass {
public static void main(String[ ] args) {
Vehicle v = new Vehicle("Blue");
}
}
This will call the constructor, which will set the color attribute to "Blue".

Constructors

A single class can have multiple constructors with different numbers of


parameters.
The setter methods inside the constructors can be used to set the
attribute values.

Example:public class Vehicle {


private String color;

Vehicle() {
this.setColor("Red");
}
Vehicle(String c) {
this.setColor(c);
}

// Setter
public void setColor(String c) {
this.color = c;
}
}
The class above has two constructors, one without any parameters setting
the color attribute to a default value of "Red", and another constructor that
accepts a parameter and assigns it to the attribute.

Now, we can use the constructors to create objects of our class.


//color will be "Red"
Vehicle v1 = new Vehicle();
//color will be "Green"
Vehicle v2 = new Vehicle("Green"); Try It Yourself

Java automatically provides a default constructor, so all classes have


a constructor, whether one is specifically defined or not.

1/1

The Math Class

The JDK defines a number of useful classes, one of them being


the Math class, which provides predefined methods for mathematical
operations.
You do not need to create an object of the Math class to use it. To access
it, just type in Math. and the corresponding method.

Math.abs() returns the absolute value of its parameter.


int a = Math.abs(10); // 10
int b = Math.abs(-20); // 20Try It Yourself

Math.ceil() rounds a floating point value up to the nearest integer value.


The rounded value is returned as a double.
double c = Math.ceil(7.342); // 8.0Try It Yourself

Similarly, Math.floor() rounds a floating point value down to the


nearest integer value.
double f = Math.floor(7.343); // 7.0Try It Yourself

Math.max() returns the largest of its parameters.


int m = Math.max(10, 20); // 20Try It Yourself

Conversely, Math.min() returns the smallest parameter.


int m = Math.min(10, 20); // 10Try It Yourself

Math.pow() takes two parameters and returns the first parameter raised
to the power of the second parameter.
double p = Math.pow(2, 3); // 8.0Try It Yourself
There are a number of other methods available in the Math class,
including:
sqrt() for square root, sin() for sine, cos() for cosine, and others.

Static

When you declare a variable or a method as static, it belongs to the class,


rather than to a specific instance. This means that only one instance of
a static member exists, even if you create multiple objects of the class, or
if you don't create any. It will be shared by all objects.
Example:public class Counter {
public static int COUNT=0;
Counter() {
COUNT++;
}
}
The COUNT variable will be shared by all objects of that class.
Now, we can create objects of our Counter class in main, and access
the static variable.
public class MyClass {
public static void main(String[ ] args) {
Counter c1 = new Counter();
Counter c2 = new Counter();
System.out.println(Counter.COUNT);
}
}
//Outputs "2"Try It Yourself

The output is 2, because the COUNT variable is static and gets


incremented by one each time a new object of the Counter class is
created. In the code above, we created 2 objects.
You can also access the static variable using any object of that class, such
as c1.COUNT.
Its a common practice to use upper case when naming a static variable,
although not mandatory.

Static

When you declare a variable or a method as static, it belongs to the class,


rather than to a specific instance. This means that only one instance of
a static member exists, even if you create multiple objects of the class, or
if you don't create any. It will be shared by all objects.
Example:public class Counter {
public static int COUNT=0;
Counter() {
COUNT++;
}
}
The COUNT variable will be shared by all objects of that class.
Now, we can create objects of our Counter class in main, and access
the static variable.
public class MyClass {
public static void main(String[ ] args) {
Counter c1 = new Counter();
Counter c2 = new Counter();
System.out.println(Counter.COUNT);
}
}
//Outputs "2"Try It Yourself

The output is 2, because the COUNT variable is static and gets


incremented by one each time a new object of the Counter class is
created. In the code above, we created 2 objects.
You can also access the static variable using any object of that class, such
as c1.COUNT.
Its a common practice to use upper case when naming a static variable,
although not mandatory.

Also, the main method must always be static.

final

Use the final keyword to mark a variable constant, so that it can be


assigned only once.
Example:
class MyClass {
public static final double PI = 3.14;
public static void main(String[ ] args) {
System.out.println(PI);
}
}Try It Yourself

PI is now a constant. Any attempt to assign it a value will cause an error.


Methods and classes can also be marked final. This serves to restrict
methods so that they can't be overridden and classes so that they can't be
made subclasses.
These concepts will be covered in the next module.
Packages

Packages are used to avoid name conflicts and to control access to


classes.
A package can be defined as a group made up of similar types of classes,
along with sub-packages.
Creating a package in Java is quite easy. Simply right click on
your src directory and click New->Package. Give your package a name
and click Finish.
You will notice that the new package appears in the project directory. Now
you can move and create classes inside that package. We have moved
our Vehicle, Counter and Animal classes to the package samples.

When you move/create a class in your package, the following code will
appear at the top of the list of files.package samples;
This indicates the package to which the class belongs.
Now, we need to import the classes that are inside a package in our main
to be able to use them.
The following example shows how to use the Vehicle class of
the samples package.import samples.Vehicle;

class MyClass {
public static void main(String[ ] args) {
Vehicle v1 = new Vehicle();
v1.horn();
}
}
Two major results occur when a class is placed in a package. First, the
name of the packagebecomes a part of the name of the class.
Second, the name of the package must match the directory structure
where the corresponding class file resides.
Use a wildcard to import all classes in a package.
For example, import samples.* will import all classes in the
samples package.

Encapsulation

There are 4 core concepts in


OOP: encapsulation, inheritance, polymorphism, and abstraction.

The idea behind encapsulation is to ensure that implementation details


are not visible to users. The variables of one class will be hidden from the
other classes, accessible only through the methods of the current class.
This is called data hiding.
To achieve encapsulation in Java, declare the class' variables
as private and provide public setterand getter methods to modify and
view the variables' values.

For example:class BankAccount {


private double balance=0;
public void deposit(double x) {
if(x > 0) {
balance += x;
}
}
}
This implementation hides the balance variable, enabling access to it only
through the deposit method, which validates the amount to be deposited
before modifying the variable.
In summary, encapsulation provides the following benefits:
- Control of the way data is accessed or modified
- More flexible and easily changed code
- Ability to change one part of the code without affecting other parts

Encapsulation

There are 4 core concepts in


OOP: encapsulation, inheritance, polymorphism, and abstraction.
The idea behind encapsulation is to ensure that implementation details are
not visible to users. The variables of one class will be hidden from the
other classes, accessible only through the methods of the current class.
This is called data hiding.
To achieve encapsulation in Java, declare the class' variables
as private and provide public setterand getter methods to modify and view
the variables' values.

For example:class BankAccount {


private double balance=0;
public void deposit(double x) {
if(x > 0) {
balance += x;
}
}
}
This implementation hides the balance variable, enabling access to it only
through the deposit method, which validates the amount to be deposited
before modifying the variable.
In summary, encapsulation provides the following benefits:
- Control of the way data is Inheritance

Inheritance is the process that enables one class to acquire the


properties (methods and variables) of another. With inheritance, the
information is placed in a more manageable, hierarchical order.

The class inheriting the properties of another is the subclass (also called
derived class, or child class); the class whose properties are inherited is
the superclass (base class, or parent class).

To inherit from a class, use the extends keyword.


This example shows how to have the class Dog to inherit from the
class Animal.class Dog extends Animal {
// some code
}
Here, Dog is the subclass, and Animal is the superclass.accessed or
modified
- More flexible and easily changed code
- Ability to change one part of the code without affecting other parts
Inheritance

When one class is inherited from another class, it inherits all of the
superclass' non-private variables and methods.
Example:class Animal {
protected int legs;
public void eat() {
System.out.println("Animal eats");
}
}

class Dog extends Animal {


Dog() {
legs = 4;
}
}
As you can see, the Dog class inherits the legs variable from the Animal
class.
We can now declare a Dog object and call the eat method of its
superclass:
class MyClass {
public static void main(String[ ] args) {
Dog d = new Dog();
d.eat();
}
}Try It Yourself

Recall the protected access modifier, which makes the members visible
only to the subclasses.

Inheritance

Constructors are not member methods, and so are not inherited by


subclasses.
However, the constructor of the superclass is called when the subclass is
instantiated.
Example:
class A {
public A() {
System.out.println("New A");
}
}
class B extends A {
public B() {
System.out.println("New B");
}
}

class Program {
public static void main(String[ ] args) {
B obj = new B();
}
}

/*Outputs
"New A"
"New B"
*/Try It Yourself

You can access the superclass from the subclass using


the super keyword.
For example, super.var accesses the var member of the superclass.

Polymorphism

Polymorphism, which refers to the idea of "having many forms", occurs


when there is a hierarchy of classes related to each other
through inheritance.
A call to a member method will cause a different implementation to be
executed, depending on the type of the object invoking the method.
Here is an example: Dog and Cat are classes that inherit from
the Animal class. Each class has its own implementation of
the makeSound() method.class Animal {
public void makeSound() {
System.out.println("Grr...");
}
}
class Cat extends Animal {
public void makeSound() {
System.out.println("Meow");
}
}
class Dog extends Animal {
public void makeSound() {
System.out.println("Woof");
}
}
As all Cat and Dog objects are Animal objects, we can do the following
in main:public static void main(String[ ] args) {
Animal a = new Dog();
Animal b = new Cat();
}
We've created two reference variables of type Animal, and pointed them to
the Cat and Dog objects.
Now, we can call the makeSound() methods.
a.makeSound();
//Outputs "Woof"

b.makeSound();
//Outputs "Meow"Try It Yourself

As the reference variable a contains a Dog object, the


makeSound() method of the Dog class will be called.
The same applies to the b variable.
This demonstrates that you can use the Animal variable without actually
knowing that it contains an object of the subclass.
This is very useful when you have multiple subclasses of the superclass.

Method Overriding

As we saw in the previous lesson, a subclass can define a behavior that's specific
to the subclass type, meaning that a subclass can implement a parent
class method based on its requirement.
This feature is known as method overriding.
Example:

class Animal {
public void makeSound() {
System.out.println("Grr...");
}
}
class Cat extends Animal {
public void makeSound() {
System.out.println("Meow");
}
}Try It Yourself

In the code above, the Cat class overrides the makeSound() method of its
superclass Animal.

Rules for Method Overriding:


- Should have the same return type and arguments
- The access level cannot be more restrictive than the overridden method's
access level (Example: If the superclass method is declared public, the
overriding method in the sub class can be neither private or protected)
- A method declared final or static cannot be overridden
- If a method cannot be inherited, it cannot be overridden
- Constructors cannot be overridden
Method overriding is also known as runtime polymorphism.

Method Overloading

When methods have the same name, but different parameters, it is known
as method overloading.
This can be very useful when you need the same method functionality for
different types of parameters.
The following example illustrates a method that returns the maximum of its two
parameters.int max(int a, int b) {
if(a > b) {
return a;
}
else {
return b;
}
}
The method shown above will only work for parameters of type integer.
However, we might want to use it for doubles, as well. For that, you need to
overload the max method:

double max(double a, double b) {


if(a > b) {
return a;
}
else {
return b;
}
}Try It Yourself

Now, our max method will also work with doubles.


An overloaded method must have a different argument list; the parameters
should differ in their type, number, or both.
Another name for method overloading is compile-time polymorphism.

Abstraction
Data abstraction provides the outside world with only essential
information, in a process of representing essential features without
including implementation details.
A good real-world example is a book. When you hear the term book, you
don't know the exact specifics, such as the page count, the color, or the
size, but you understand the idea, or abstraction, of a book.
The concept of abstraction is that we focus on essential qualities, rather
than the specific characteristics of one particular example.

In Java, abstraction is achieved using abstract classes and interfaces.


An abstract class is defined using the abstract keyword.
- If a class is declared abstract it cannot be instantiated (you cannot create
objects of that type).
- To use an abstract class, you have to inherit it from another class.
- Any class that contains an abstract method should be defined as
abstract.
An abstract method is a method that is declared without an
implementation (without braces, and followed by a
semicolon): abstract void walk();

Abstract Class

For example, we can define our Animal class as abstract: abstract class Animal {
int legs = 0;
abstract void makeSound();
}
The makeSound method is also abstract, as it has no implementation in the
superclass.
We can inherit from the Animal class and define the makeSound() method for the
subclass:

class Cat extends Animal {


public void makeSound() {
System.out.println("Meow");
}
}Try It Yourself

Every Animal makes a sound, but each has a different way to do it. That's why
we define an abstract class Animal, and leave the implementation of how they
make sounds to the subclasses.
This is used when there is no meaningful definition for the method in the
superclass.

Interfaces

An interface is a completely abstract class that contains only abstract


methods.
Some specifications for interfaces:
- Defined using the interface keyword.
- May contain only static final variables.
- Cannot contain a constructor because interfaces cannot be instantiated.
- Interfaces can extend other interfaces.
- A class can implement any number of interfaces.

An example of a simple interface:interface Animal {


public void eat();
public void makeSound();
}
Interfaces have the following properties:
- An interface is implicitly abstract. You do not need to use the abstract
keyword while declaring an interface.
- Each method in an interface is also implicitly abstract, so the abstract
keyword is not needed.
- Methods in an interface are implicitly public.
A class can inherit from just one superclass, but can
implement multiple interfaces!

Interfaces

Use the implements keyword to use an interface with your class.

interface Animal {
public void eat();
public void makeSound();
}

class Cat implements Animal {


public void makeSound() {
System.out.println("Meow");
}
public void eat() {
System.out.println("omnomnom");
}
}Try It Yourself

When you implement an interface, you need to override all of its methods.

Type Casting

Assigning a value of one type to a variable of another type is known as Type


Casting.

To cast a value to a specific type, place the type in parentheses and position it in
front of the value.
Example:

int a = (int) 3.14;


System.out.println(a);
//Outputs 3Try It Yourself

The code above is casting the value 3.14 to an integer, with 3 as the resulting
value.
Another example:

double a = 42.571;
int b = (int) a;
System.out.println(b);
//Outputs 42Try It Yourself

Java supports automatic type casting of integers to floating points, since there is
no loss of precision.
On the other hand, type casting is mandatory when assigning floating point
values to integer variables.

Type Casting

For classes, there are two types of casting.


Upcasting

You can cast an instance of a subclass to its superclass.


Consider the following example, assuming that Cat is a subclass of
Animal.Animal a = new Cat();
Java automatically upcasted the Cat type variable to the Animal type.

Downcasting

Casting an object of a superclass to its subclass is called downcasting.


Example:Animal a = new Animal();
((Cat)a).makeSound();
This will try to cast the variable a to the Cat type and call its
makeSound() method.
Why is upcasting automatic, downcasting manual? Well, upcasting can never fail.
But if you have a group of different Animals and want to downcast them all to a
Cat, then there's a chance that some of these Animals are actually Dogs, so the
process fails.

Anonymous Classes

Anonymous classes are a way to extend the existing classes on the fly.
For example, consider having a class Machine:class Machine {
public void start() {
System.out.println("Starting...");
}
}
When creating the Machine object, we can change the start method on the fly.

public static void main(String[ ] args) {


Machine m = new Machine() {
@Override public void start() {
System.out.println("Wooooo");
}
};
m.start();
}
//Outputs "Wooooo";Try It Yourself

After the constructor call, we have opened the curly braces and have overridden
the start method's implementation on the fly.
The @Override annotation is used to make your code easier to understand,
because it makes it more obvious when methods are overridden.

Anonymous Classes

The modification is applicable only to the current object, and not the class
itself. So if we create another object of that class, the start method's
implementation will be the one defined in the class.
class Machine {
public void start() {
System.out.println("Starting...");
}
}
public static void main(String[ ] args) {
Machine m1 = new Machine() {
@Override public void start() {
System.out.println("Wooooo");
}
};
Machine m2 = new Machine();
m2.start();
}
//Outputs "Starting..."

Inner Classes

Java supports nesting classes; a class can be a member of another class.


Creating an inner class is quite simple. Just write a class within a class. Unlike a
class, an inner class can be private. Once you declare an inner class private, it
cannot be accessed from an object outside the class.
Example:

class Robot {
int id;
Robot(int i) {
id = i;
Brain b = new Brain();
b.think();
}

private class Brain {


public void think() {
System.out.println(id + " is thinking");
}
}

}Try It Yourself

The class Robot has an inner class Brain. The inner class can access all of the
member variables and methods of its outer class, but it cannot be accessed from
any outside class.

Comparing Objects

Remember that when you create objects, the variables store references to the
objects.
So, when you compare objects using the equality testing operator (==), it
actually compares the references and not the object values.
Example:

class Animal {
String name;
Animal(String n) {
name = n;
}
}

class MyClass {
public static void main(String[ ] args) {
Animal a1 = new Animal("Robby");
Animal a2 = new Animal("Robby");
System.out.println(a1 == a2);
}
}
//Outputs falseTry It Yourself

Despite having two objects with the same name, the equality testing returns
false, because we have two different objects (two different references or memory
locations).

equals()
Each object has a predefined equals() method that is used for semantical
equality testing.
But, to make it work for our classes, we need to override it and check the
conditions we need.
There is a simple and fast way of generating the equals() method, other than
writing it manually.
Just right click in your class, go to Source->Generate hashCode() and
equals()...

This will automatically create the necessary methods. class Animal {


String name;
Animal(String n) {
name = n;
}
@Override
public int hashCode() {
final int prime = 31;
int result = 1;
result = prime * result + ((name == null) ? 0 : name.hashCode());
return result;
}
@Override
public boolean equals(Object obj) {
if (this == obj)
return true;
if (obj == null)
return false;
if (getClass() != obj.getClass())
return false;
Animal other = (Animal) obj;
if (name == null) {
if (other.name != null)
return false;
} else if (!name.equals(other.name))
return false;
return true;
}
}
The automatically generated hashCode() method is used to determine where to
store the object internally. Whenever you implement equals, you MUST also
implement hashCode.
We can run the test again, using the equals method:

public static void main(String[ ] args) {


Animal a1 = new Animal("Robby");
Animal a2 = new Animal("Robby");
System.out.println(a1.equals(a2));
}
//Outputs trueTry It Yourself

You can use the same menu to generate other useful methods, such
as getters and setters for your class attributes.

Enums

You should always use Enums when a variable (especially


a method parameter) can only take one out of a small set of possible
values.
If you use Enums instead of integers (or String codes), you increase
compile-time checking and avoid errors from passing in invalid constants,
and you document which values are legal to use.
Some sample Enum uses include month names, days of the week, deck of
cards, etc.

Java API

The Java API is a collection of classes and interfaces that have been
written for you to use.
The Java API Documentation with all of the available APIs can be located
on the Oracle website at
http://docs.oracle.com/javase/7/docs/api/
Once you locate the package you want to use, you need to import it into
your code.
The package can be imported using the import keyword.
For example:import java.awt.*;
The awt package contains all of the classes for creating user interfaces
and for painting graphics and images.
The wildcard character (*) is used to import all of the classes in
the package.

Exceptions

An exception is a problem that occurs during program execution.


Exceptions cause abnormal termination of the program.
Exception handling is a powerful mechanism that handles runtime errors
to maintain normal application flow.

An exception can occur for many different reasons. Some examples:


- A user has entered invalid data.
- A file that needs to be opened cannot be found.
- A network connection has been lost in the middle of communications.
- Insufficient memory and other issues related to physical resources.
As you can see, exceptions are caused by user error, programmer error, or
physical resource issues. However, a well-written program should handle
all possible exceptions.

Exception Handling

Exceptions can be caught using a combination of the try and catch keywords.
A try/catch block is placed around the code that might generate an exception.
Syntax:try {
//some code
} catch (Exception e) {
//some code to handle errors
}
A catch statement involves declaring the type of exception you are trying to
catch. If an exceptionoccurs in the try block, the catch block that follows
the try is checked. If the type of exception that occurred is listed in
a catch block, the exception is passed to the catch block much as
an argument is passed into a method parameter.
The Exception type can be used to catch all possible exceptions.
The example below demonstrates exception handling when trying to access
an array index that does not exist:

public class MyClass {


public static void main(String[ ] args) {
try {
int a[ ] = new int[2];
System.out.println(a[5]);
} catch (Exception e) {
System.out.println("An error occurred");
}
}
}
//Outputs "An error occurred"Try It Yourself

Without the try/catch block this code should crash the program, as a[5] does not
exist.
Notice the (Exception e) statement in the catch block - it is used to catch all
possible Exceptions.

throw

The throw keyword allows you to manually generate exceptions from your
methods. Some of the numerous available exception types include the
IndexOutOfBoundsException, IllegalArgumentException, ArithmeticException, and
so on.
For example, we can throw an ArithmeticException in our method when the
parameter is 0.

int div(int a, int b) throws ArithmeticException {


if(b == 0) {
throw new ArithmeticException("Division by Zero");
} else {
return a / b;
}
}Try It Yourself

The throws statement in the method definition defines the type of Exception(s)
the method can throw.
Next, the throw keyword throws the corresponding exception, along with a
custom message.
If we call the div method with the second parameter equal to 0, it will throw an
ArithmeticException with the message "Division by Zero".
Multiple exceptions can be defined in the throws statement using a comma-
separated list.

Threads

Java is a multi-threaded programming language. This means that our program


can make optimal use of available resources by running two or more components
concurrently, with each component handling a different task.
You can subdivide specific operations within a single application into
individual threads that all run in parallel.
The following diagram shows the life-cycle of a thread.

There are two ways to create a thread.


1. Extend the Thread class
Inherit from the Thread class, override its run() method, and write the
functionality of the thread in the run() method.
Then you create a new object of your class and call it's start method to run the
thread.
Example:
class Loader extends Thread {
public void run() {
System.out.println("Hello");
}
}

class MyClass {
public static void main(String[ ] args) {
Loader obj = new Loader();
obj.start();
}
}Try It Yourself

As you can see, our Loader class extends the Thread class and overrides
its run() method.
When we create the obj object and call its start() method,
the run() method statements execute on a different thread.
Every Java thread is prioritized to help the operating system determine the order
in which to schedule threads. The priorities range from 1 to 10, with each thread
defaulting to priority 5. You can set the thread priority with
the setPriority() method.

Threads

The other way of creating Threads is implementing the Runnable interface.


Implement the run() method. Then, create a new Thread object, pass the
Runnable class to its constructor, and start the Thread by calling
the start() method.
Example:

class Loader implements Runnable {


public void run() {
System.out.println("Hello");
}
}

class MyClass {
public static void main(String[ ] args) {
Thread t = new Thread(new Loader());
t.start();
}
}Try It Yourself

The Thread.sleep() method pauses a Thread for a specified period of time. For
example, calling Thread.sleep(1000); pauses the thread for one second. Keep
in mind that Thread.sleep() throws an InterruptedException, so be sure to
surround it with a try/catch block.
It may seem that implementing the Runnable interface is a bit more complex
than extending from the Thread class. However, implementing the
Runnable interface is the preferred way to start a Thread, because it enables you
to extend from another class, as well.

Types of Exceptions

There are two exception types, checked and unchecked (also called
runtime). The main difference is that checked exceptions are checked
when compiled, while unchecked exceptions are checked at runtime.
As mentioned in our previous lesson, Thread.sleep() throws an
InterruptedException. This is an example of a checked exception. Your
code will not compile until you've handled the exception.

public class MyClass {


public static void main(String[ ] args) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
//some code
}
}
}
We have seen examples of unchecked exceptions, which are checked at
runtime, in previous lessons. Example (when attempting to divide by 0):
public class MyClass {
public static void main(String[ ] args) {
int value = 7;
value = value / 0;
}
}
/*
Exception in thread "main" java.lang.ArithmeticException: / by zero
at MyClass.main(MyClass.java:4)
*/

ArrayList

The Java API provides special classes to store and manipulate groups of
objects.
One such class is the ArrayList. Standard Java arrays are of a fixed length,
which means that after they are created, they cannot expand or shrink.
On the other hand, ArrayLists are created with an initial size, but when
this size is exceeded, the collection is automatically enlarged.
When objects are removed, the ArrayList may shrink in size. Note that
the ArrayList class is in the java.util package, so it's necessary to import
it before using it.
Create an ArrayList as you would any object.import java.util.ArrayList;
//...
ArrayList colors = new ArrayList();
You can optionally specify a capacity and type of objects
the ArrayList will hold:ArrayList<String> colors = new
ArrayList<String>(10);
The code above defines an ArrayList of Strings with 10 as its initial size.
ArrayLists store objects. Thus, the type specified must be a class type. You
cannot pass, for example, int as the objects' type. Instead, use the
special class types that correspond to the desired value type, such
as Integer for int, Double for double, and so on.

ArrayList

The ArrayList class provides a number of useful methods for manipulating its
objects.
The add() method adds new objects to the ArrayList. Conversely, the remove()
methods remove objects from the ArrayList.
Example:
import java.util.ArrayList;

public class MyClass {


public static void main(String[ ] args) {
ArrayList<String> colors = new ArrayList<String>();
colors.add("Red");
colors.add("Blue");
colors.add("Green");
colors.add("Orange");
colors.remove("Green");

System.out.println(colors);
}
}
// Output: [Red, Blue, Orange]Try It Yourself

Other useful methods include the following:


- contains(): Returns true if the list contains the specified element
- get(int index): Returns the element at the specified position in the list
- size(): Returns the number of elements in the list
- clear(): Removes all of the elements from the list

Note: As with arrays, the indexing starts with 0

You cannot specify an initial capacity for the LinkedList.

LinkedList vs. ArrayList

The most notable difference between the LinkedList and the ArrayList is in the
way they store objects.
The ArrayList is better for storing and accessing data, as it is very similar to a
normal array.
The LinkedList is better for manipulating data, such as making numerous
inserts and deletes.

In addition to storing the object, the LinkedList stores the memory address (or
link) of the element that follows it. It's called a LinkedList because each element
contains a link to the neighboring element.
You can use the enhanced for loop to iterate over its elements.

LinkedList<String> c = new LinkedList<String>();


c.add("Red");
c.add("Blue");
c.add("Green");
c.add("Orange");
c.remove("Green");

for(String s: c) {
System.out.println(s);
}
/* Output:
Red
Blue
Orange
*/Try It Yourself

Summary:
- Use an ArrayList when you need rapid access to your data.
- Use a LinkedList when you need to make a large number of inserts and/or
deletes.

HashMap

Arrays and Lists store elements as ordered collections, with each element given
an integer index.
HashMap is used for storing data collections as key and value pairs. One object
is used as a key (index) to another object (the value).
The put, remove, and get methods are used to add, delete, and access values
in the HashMap.
Example:

import java.util.HashMap;
public class MyClass {
public static void main(String[ ] args) {
HashMap<String, Integer> points = new HashMap<String, Integer>();
points.put("Amy", 154);
points.put("Dave", 42);
points.put("Rob", 733);
System.out.println(points.get("Dave"));
}
}
// Outputs 42Try It Yourself

We have created a HashMap with Strings as its keys and Integers as its values.
Use the get method and the corresponding key to access
the HashMap elements.

HashMap

Arrays and Lists store elements as ordered collections, with each element given
an integer index.
HashMap is used for storing data collections as key and value pairs. One object
is used as a key (index) to another object (the value).
The put, remove, and get methods are used to add, delete, and access values
in the HashMap.
Example:

import java.util.HashMap;
public class MyClass {
public static void main(String[ ] args) {
HashMap<String, Integer> points = new HashMap<String, Integer>();
points.put("Amy", 154);
points.put("Dave", 42);
points.put("Rob", 733);
System.out.println(points.get("Dave"));
}
}
// Outputs 42Try It Yourself

We have created a HashMap with Strings as its keys and Integers as its values.
Use the get method and the corresponding key to access
the HashMap elements.

You might also like