Java Data Types

Learn about various data types in Java. Learn the differences between primitive datatypes and non-primitive datatypes (or reference datatype). We will also learn about the data types sizes and best practices to use datatypes in Java.

1. Variables and Data Types

In Java, typically data-types are associated with variables. A variable declaration has three parts:

  1. A variable name (also called identifier) to refer to the memory location
  2. The variable type stored at the memory location (it is called datatype)
  3. A memory location to hold the value of the variable
Java Data Type
Java Data Type

The second property (marked in red) is called data type. The data type of the variable determines the range of the values that the memory location can hold.

Therefore, the amount of memory allocated for a variable depends on its data type. For example, 32 bits of memory is allocated for a variable of the 'int' data type.

Java is a statically-typed language. This means all variables MUST be declared before they can be used.

boolean flag = true;

int counter = 20;

2. Data types in Java

Java supports two kinds of data types i.e. primitive data type and non-primitive or reference data type.

2.1. Primitive Data Types

A primitive data type directly holds a value in memory. For instance, a number or a character. Primitive data types are not objects, as well as not references to the objects.

The values stored in primitives are called literals. A literal is the source code representation of a fixed value; literals are represented directly in your code without requiring computation.

In Java, we have eight types of primitive data types.

Data TypeDescriptionDefault ValueMemory Size
booleanA binary value of either true or falsefalse1 bit
charAny unicode character\u0000 (0)16 bit unicode character
bytevalues values from -128 to 12708 bit signed value
shortvalues values from -32768 to 32767016 bits signed value
intvalues values from from -231 to 231-1032 bits signed value
longvalues values from from -263 to 263-1064 bit floating point value
floatIEEE 754 floating point0.032 bit floating point value
doubleIEEE 754 floating point0.064 bit floating point value

In Java SE 7 and later, any number of underscore characters ('_') can appear anywhere between digits in a numerical literal. e.g. 10_000_000 is a valid number in Java.

Type conversion between primitives

Except boolean, we can assign a primitive value to another primitive type. Though, sometimes it may result in data loss when a primitive of large memory capacity is assigned to primitive with smaller memory capacity.

It’s just like you are transferring the water from a large vessel and putting in a smaller vessel, so the loss of water is natural.

int counter = 20_000_000;
        
//Assign int to short (data loss)
short shortCounter = (short) counter;

//assign int to long (no data loss)
long longCounter = counter;

System.out.println(counter);            //20000000
System.out.println(shortCounter);       //11520
System.out.println(longCounter);        //20000000

Notice that when Java detects that type conversion may result in data loss (bigger data type to smaller one), then gives type-mismatch error and explicitly asks for type casting (e.g. ‘int’ to ‘short’ assignment). It helps in detecting and resolving accidental data loss assignments.

2.2. Non-primitive Data Types

A non-primitive or reference data type holds the reference to an object in memory. Using the reference stored in the variable, you can access fields and methods of the referenced object.

For example, java.lang.String is a class defined in the Java library and you can use it to manipulate text (sequence of characters). You declare a reference variable of type String as:

String str = new String( "Hello World !!" );

What happens when this code is executed? First, a memory block is allocated, and the name of the variable str is associated with that memory location. This process is the same as declaring a primitive data type variable.

The second part of code creates a new String object in memory with text "Hi" and stores the reference (or memory address) of the String object into the variable 'str'.

Multiple references can refer to same object

You can also assign the reference of an object stored in one reference variable to another reference variable. In such cases, both reference variables will refer to the same object in memory.

// Declares String reference variable str1 and str2
String str1;
String str2;
 
// Assigns the reference of a String object "Hello" to str1
str1 = new String( "Hello World !!" );
 
// Assigns the reference stored in str1 to str2
str2 = str1;

System.out.println( str1 );         //Hello World !!
System.out.println( str2 );         //Hello World !!
There is a reference constant (also known as reference literal) null, which can be assigned to any reference variable. If null is assigned to a reference variable, it means that the reference variable is not referring to any object in memory.

3. Wrapper Classes

A wrapper class is a class whose object wraps or contains primitive data types. In other words, we can wrap a primitive value into a wrapper class object.

Please note that Java has one wrapper class mapped to each primitive data type. For example, java.lang.Integer class is object version of int data type. Similarly, we have total 8 wrapper classes for all 8 primitive data types.

The wrapper class names are the same as primitive data types names, only starting with a capital letter.

These wrapper classes are Boolean, Byte, Short, Character, Integer, Long, Float and Double.

4. Auto-boxing

In Java, you can assign a primitive type value to a wrapper class, directly. For example, you can assign a int value to Interger class reference.

Integer counter = 20;

static Float PI = 3.14f;

It’s worth mentioning that all wrapper class instances are immutable. They also maintain an internal cache for performance reason.

5. Difference between primitive and non-primitive data types

  1. Primitives store values directly, which are called literals. Reference types store references to actual objects in memory area.
  2. There are 8 fixed primitive data types. In Java, each class is a data type including wrapper classes.

6. Best practices

  1. Use Java variable naming conventions and follow best practices.
  2. Use primitives for variables which are local in scope. e.g. inside methods, counter for loops and intermediate results.
  3. When data is transferred among method or classes, it’s better to use objects because only their references will be copied, no memory overhead will be added.
  4. When dealing with collections (which need objects), you shall be using Objects.
  5. While sending data over network, use objects and make them Serializable. Wrapper classes are automatically Serializable.
  6. Always know the size of data type you will need. Use appropriate data sizes. Using int to store boolean values (0 and 1) is waste of memory.
  7. Use underscores (above Java 7) in numbers. It make them more readable.

Happy Learning !!

Was this post helpful?

Join 7000+ Fellow Programmers

Subscribe to get new post notifications, industry updates, best practices, and much more. Directly into your inbox, for free.

Leave a Comment

HowToDoInJava

A blog about Java and its related technologies, the best practices, algorithms, interview questions, scripting languages, and Python.