DEV Community

Cover image for What are Java Collections? Get started with the framework
Erin Schaffer for Educative

Posted on • Originally published at educative.io

What are Java Collections? Get started with the framework

Collections are used in every programming language. They are objects that group multiple elements into a single unit. Before the Collections Framework, it was hard for programmers to write algorithms that worked for different kinds of collections. Java came with some Collection classes, like Vector, Stack, Hashtable, and Array, but they had their disadvantages.

In JDK 1.2, Java developers introduced the Collections Framework, which is an important framework to help you achieve all of your data operations. Today, we’ll dive deeper into this framework and some of its components.

Here’s what we’ll cover today:

What are Java Collections?

The Java Collections Framework is a unified architecture for representing and manipulating collections. It contains interfaces, their implementation classes, and algorithms to process the data stored in a collection. The Collection interface is extended by other interfaces like List, Set, and Queue.

Note: There’s also a Map interface, but it doesn’t implement the Collection interface because it stores key-value pairs, and the classes that come under the Collection interface only store values.

Differences between the Collections Framework and Collections:

  • The Collections Framework is an interface, while Collections is a class.
  • The Collection interface gives the standard functionality of data structures to List, Set, and Queue. The Collections class provides standard methods that you can use to search, sort, and coordinate collection elements.
  • Collections are objects that represent a group of objects (like Vector), while the Collections Framework can represent and manipulate collections.

The Java Collections Framework components

Interfaces

These interfaces supply the abstract data type to represent the collection. The java.util.Collection is the root interface of the framework. It's at the top of the framework hierarchy and contains important methods, like size(), iterator(), add(), remove(), and clear().

The iterable interface is the root of the whole collection framework. It allows the iterator to iterate through all of the collections. All classes and interfaces use this interface. The collection interface extends the iterable interface and is executed by the classes in the collection framework. The list interface inhibits a list-type data structure where we can store ordered collections of objects.

Some more important interfaces include:

  • Map interface: java.util.Map
  • Set interface: java.util.Set
  • Deque interface: java.util.Deque

Note: The Map interface is the only one that doesn't inherit from the Collection interface, but is included in the Collections Framework. All framework interfaces are in the java.util package.

Implementation classes

The framework provides implementation classes for collections. You can use them to create different types of collections in your Java programs.
Some of the main collection classes include:

  • ArrayList
  • LinkedList
  • PriorityQueue
  • HashMap and HashSet
  • TreeMap and TreeSet

Algorithms

These algorithms perform important functions on collections, like sorting lists.

The Collections Framework hierarchy is as follows:

Alt Text



ArrayList in Java

ArrayList is the most commonly used implementation of the List interface. Some of its features include:

  • Stores elements in insertion order
  • Allows for storage of duplicate elements
  • Supports null elements

An ArrayList stores data in a resizable array. When you create an ArrayList, you create an array of size zero. When the first element is inserted, the array size changes to ten. This is known as lazy initialization and it saves a lot of memory. Before adding an element to an ArrayList, the capacity is checked. If the array is full, then a new array of size (n + n/2 + 1) is created. The elements from the old array are then copied to the new one.

There are three ways to create an ArrayList in Java:

1. The no-arg constructor

This constructor doesn't take any arguments and will create a list size of zero.

List list = newArrayList();
Enter fullscreen mode Exit fullscreen mode

2. The constructor that takes initial capacity

You can give an initial capacity when creating an ArrayList. Let's say you know that your ArrayList will contain a minimum of 50 elements. You can create your ArrayList with a size of 50, reducing the need for constant resizing.

List list = newArrayList(50);
Enter fullscreen mode Exit fullscreen mode

3. Existing Collection

You can create an ArrayList using an existing Collection, and the list will contain all the elements in the original collection in the same order.

List list = newArrayList(oldList);
Enter fullscreen mode Exit fullscreen mode

Here's an example of an ArrayList you could execute in Java:

import java.util.ArrayList;
import java.util.List;
public class ArrayListDemo {
    public static void main(String args[]) {
        List<Integer> list = new ArrayList<>();
        list.add(1);
        list.add(2);
        list.add(3);
        System.out.println(list);
        System.out.println("The element at index two is " + list.get(1));
        System.out.println("The size of the List is " + list.size());
    }
}
Enter fullscreen mode Exit fullscreen mode

Output:

[1, 2, 3]
The element at index two is 2
The size of the List is 3

There are many things you can do with an ArrayList, including:

  • Update an element using the set(int index, E e) method
  • Check if an element is present using the contains(Object o) method
  • Remove all elements within a given Collection using the removeAll(Collection<?> c) method
  • Replace all elements using the replaceAll(UnaryOperator<E> operator)
  • Etc.

LinkedList in Java

The LinkedList class implements the List and Deque interfaces. Some of its features include:

  • Stores elements in insertion order
  • Supports duplicate elements
  • Allows any number of null elements

LinkedList also has a static inner class called Node, which contains three fields:

  1. item: contains the value of the current element
  2. next: contains the pointer to the next element
  3. prev: contains the pointer to the previous element]

When an element is added to a LinkedList, it creates a new Node instance. The prev and next fields are set depending on where the new node is added.

There are two ways to create a LinkedList in Java:

1. The no-arg constructor

This constructor doesn't take any arguments and will create a list size of zero.

List<Integer> list = new LinkedList<Integer>();
Enter fullscreen mode Exit fullscreen mode

2. Existing Collection

You can create a LinkedList using an existing Collection, and the list will contain all the elements in the original collection in the same order.

List<Integer> list = new LinkedList<Integer>(oldList);
Enter fullscreen mode Exit fullscreen mode

Here's an example of a LinkedList you could execute in Java:

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;
public class LinkedListDemo {
    public static void main(String args[]) {
        LinkedList<Integer> linkedList = new LinkedList<>();
        linkedList.add(1); // Adds 1 to the list.
        linkedList.add(2); // Adds 2 to the end of the list.
        linkedList.addLast(3); // Adds 3 to the end of the list.
        System.out.println(linkedList);
        linkedList.addFirst(10); // Adds 10 to the start of the list.
        System.out.println(linkedList);
        linkedList.add(2, 20); // Adds 20 to second position in the list.
        System.out.println(linkedList);
        List<Integer> list = new ArrayList<>();
        list.add(101);
        list.add(102);
        list.add(103);
        linkedList.addAll(3, list); // Adds the collection of elements at third position in the list.
        System.out.println(linkedList);
    }
}
Enter fullscreen mode Exit fullscreen mode

Output:

[1, 2, 3]
[10, 1, 2, 3]
[10, 1, 20, 2, 3]
[10, 1, 20, 101, 102, 103, 2, 3]

Some more things you can do with LinkedList:

  • Fetch an element at a particular index using the get(int index) method
  • Remove the last element using the removeLast() method
  • Sort a LinkedList using the sort() method
  • Etc.

HashSet in Java

HashSet is a class in the java.util package which implements the Set interface. Some of the features of the HashSet include:

  • Doesn't allow duplicate elements
  • Only allows one null element
  • Elements are inserted in a random order
  • Internally backed by a HashMap

There are four ways to create a HashSet in Java:

1. The no-arg constructor

This will create a HashSet with an initial capacity of 16 and a load factor of 0.75.

Set<Integer> set= new HashSet<>();
Enter fullscreen mode Exit fullscreen mode

2. A constructor that takes initial capacity

If you know your HashSet will have more than 16 elements, you can set a higher initial capacity to reduce the need for resizing. It will use a default load factor of 0.75.

3. A constructor that takes initial capacity and load factor

You can also provide an initial load factor along with an initial capacity.

4. A constructor that takes another Set as a parameter

You can create a HashSet using another Set by passing it to the constructor. It will have the same size as the passed set and a default load factor of 0.75.

Here's an example of a HashSet you could execute in Java:

import java.util.HashSet;
import java.util.Set;
public class HashSetDemo {
    public static void main(String args[]) {
        Set<Integer> set = new HashSet<>();
        System.out.println("Inserting 17 in the HashSet:  " + set.add(17));
        System.out.println("Inserting 34 in the HashSet:  " + set.add(34));
        System.out.println("Inserting 17 in the HashSet:  " + set.add(17));
        System.out.println(set);
    }
}
Enter fullscreen mode Exit fullscreen mode

Output:

Inserting 17 in the HashSet: true
Inserting 34 in the HashSet: true
Inserting 17 in the HashSet: false
[17, 34]

The above code showed a demonstration of the add(E e) method that inserts an element into the HashSet. If the inserted element wasn't already in the HashSet, the method returned true. If the element was already in the HashSet, the method returned false.

There's more you can do with HashSets, like:

  • Remove elements using the remove(Object o) method
  • Check if the HashSet is empty using the isEmpty() method
  • Fetch an element from a HashSet using the contains() method
  • Etc.

TreeSet in Java

The TreeSet class implements the Set interface that uses a tree for storage. Some of the features include:

  • Doesn't allow duplicate elements
  • Doesn't allow null elements
  • Quick access and retrieval times
  • Elements stored in ascending order

The TreeSet hierarchy is as follows:

Alt Text

How is a TreeSet different from a HashSet?

  • HashSet allows one null element, while TreeSet doesn't allow any
  • Elements are sorted randomly in HashSet, while they are sorted in order in TreeSet
  • HashSet is faster for operations like add, remove, contains, size, etc.

There are four ways to create a TreeSet in Java:

1. The no-arg constructor

Set<Integer> set = new TreeSet<>();
Enter fullscreen mode Exit fullscreen mode

2. A constructor with Comparator as an argument

If the objects you store in your TreeSet don't implement the Comparator interface or if you need to store the elements in descending order, you can provide a custom Comparator while creating a TreeSet, which will sort the elements per the logic of the Comparator.

3. A constructor with the Collection type argument

You can create a TreeSet from another Collection. The elements will be stored in ascending order.

4. A constructor with the argument of type SortedSet

This constructor acts as a copy constructor and creates a new sorted set with the same elements and the same order of the provided sorted set.

Here's an example of a TreeSet you could execute in Java:

import java.util.Comparator;
import java.util.TreeSet;
public class TreeSetDemo {
    public static void main(String args[]) {
        TreeSet<Integer> set = new TreeSet<>();
        set.add(21);
        set.add(32);
        set.add(44);
        set.add(11);
        set.add(54);
        System.out.println("TreeSet elements in ascending order " + set);
        // This TreeSet will store the elements in reverse order.
        TreeSet<Integer> reverseSet = new TreeSet<>(Comparator.reverseOrder());
        reverseSet.add(21);
        reverseSet.add(32);
        reverseSet.add(44);
        reverseSet.add(11);
        reverseSet.add(54);
        System.out.println("TreeSet elements in descending order " + reverseSet);
    }
}
Enter fullscreen mode Exit fullscreen mode

Output:

TreeSet elements in ascending order [11, 21, 32, 44, 54]
TreeSet elements in descending order [54, 44, 32, 21, 11]

There are more things you can do with TreeSet, such as:

  • Fetch elements greater than the given element using the tailSet(E fromElement) method
  • Fetch a subset of elements using the subSet(E fromElement, E to Element) method
  • Etc.

Other collections to study

Congratulations on taking your first steps with the Java Collections Framework! Collection is one of the most important topics in Java programming, and this important framework will help you accomplish all of your data operations. You’re now ready to dive deeper into the Java Collections Framework and learn about more topics such as:

  • LinkedHashMap and LinkedHashSet
  • SortedMap and SortedSet
  • toArray()
  • hasNext()
  • Etc.

To get started learning more about the Java Collections Framework, check out Educative’s course, Collections in Java. This curated, hands-on course covers all the collection types available in the Collections Framework. You’ll cover the internal workings of each Java collection so you can work with them more efficiently. The skills you gain in this course will give you an edge in Java-based interviews.

Happy learning!

Continue reading about Java

Top comments (0)