NOTICE: LinkedList
implements both List
and Deque
. Diamond boxes are classes, rest are interfaces. Map doesn’t from the Collection
interface.
List<Integer> listArr = new ArrayList<Integer>();
List<Integer> listArr = new ArrayList<>(); // RHS generic type is inferred and is optional
ArrayList
is faster in access (linear time) but slower in writing to List, LinkedList
is vice-versa.
When we create a ArrayList without any initial capacity, it allocates 10 to it by default which means that upto 10 elements JVM doesn’t need to resize the list dynamically (by creating new array internally and assigning back to same ref). We can specify a diff initial capacity in constructor. This is different from list’s size as size is the number of actual elements stored rather than max allowed capacity value.
List<Integer> listArr = new ArrayList<>(); //empty list, size = 0; init capacity = 10
List<Integer> listArr2 = new ArrayList<>(anotherList);
List<Integer> listArr3 = new ArrayList<>(20); //empty list, size = 0; init capacity = 20
The ArrayList capacity is resized by 50% of its current size when all elements are filled. So, ArrayList will be resized from 10, to 15, to 22, to 33 and so on.
The List<>
created with constructor are mutable ofcourse because of this dynamic capacity resizing by Java.
Arrays.asList(varargs) // can only update existing elements, not add or remove additional
// changes to the returned list "write through" to the original array
List.of(varargs) // purely immutable; error on any modification
List.copyOf(collection) // purely immutable; error on any modification
var listArr = new ArrayList<>(); // assumes type to be ArrayList<Object>; bad practice because ArrayList<Integer> and ArrayList<String> are both ArrayList<Object> but not convertible with each other
add(Element) // add at end
add(index, Element) // add at index and move others towards end
set(index, Element) // replace at index
get(index)
remove(int index) // remove at a specific index
remove(T Element) // remove an element having suppied value (overloaded method; smarter)
isEmpty()
size()
clear()
contains(Element)
forEach()
equals()
Arrays.asList(nums);
// if nums array is of wrapper-class type Integer[], then above will work
// if nums array is of primitive type int[], then above will NOT work (no autoboxing for arrays, only single element)
// acts as a bridge = changes to the returned list "write through" to the original array!
Integer[] list = {1, 2, 3};
List<Integer> listWrapper = Arrays.asList(list);
listWrapper.set(1, 99);
System.out.println(list[1]); // 99
// primitive int[] has to be converted to List using a loop
List<Integer> listNums = new ArrayList<>();
for(int e : nums){
listNums.add(e); // auto-boxing
}
Object[] arr = list.toArray();
String[] arr = list.toArray(new String[0]);
// 1. copy list into a new array; an array of Object type is created by default
// 2. makes a String array; size can be anything it doesn't matter, its just there to specify type
Collections define their own type based on the generic <>
parameter type. List<String>
is not convertible to List<Integer>
, neither is it convertible to List<CharSequence>
even though String
type can be converted to its superclass type CharSequence
.
Check both “type of list” and its “contents” when checking for compatibility.
Collection Overriding Thumb Rule: generic type <>
should match exactly when overriding collections, collection type itself must be same or covariant
class X{
List<CharSequence> foo(){ }
}
class Y extends X{
@Override
List<String> foo(){ } // compiler-error
}
class X{
List<CharSequence> foo(){ }
}
class Y extends X{
@Override
ArrayList<String> foo(){ } // compiler-error; list <> parameter type is causing the issue
}
class X{
List<CharSequence> foo(){ }
}
class Y extends X{
@Override
ArrayList<CharSequence> foo(){ } // valid
}
Doesn’t store duplicate entries.
HashSet: uses Hashtable so Constant time access but loses order
Uses hashCode()
and equals()
to find corresponding bucket to put item in.
TreeSet: uses sorted tree structure (BST); logarithmic time for access but order is numerically sorted
Set<Integer> s = new HashSet<>();
s.add(1);
s.add(2);
s.add(3);
s.add(3);
for(Integer i: s)
System.out.println(i); // 2 3 1 (arbitrary)
Set<Integer> s = new TreeSet<>();
s.add(3);
s.add(2);
s.add(1);
s.add(3);
for(Integer i: s)
System.out.println(i); // 1 2 3 (numerically sorted)
All are immutable.
Set<Integer> set = new HashSet<>();
// Factory Methods
Set.of(varargs)
Set.copyOf(collection)
Use ArrayDeque
if you don’t need List methods.
Single ended FIFO queue - Queue<>
(<- remove <- ... <- add <-
) (front --- back
)
We often use queues as stacks (recommended over the legacy Stack
collection class), use Deque<>
which has push()
/pop()
and First
/Last
methods available extra which push and pop at the same side (front).
// a Deque<Integer> q -> 1 2 3
// First ---- Last
// Top --- Bottom
q.add(4); // 1 2 3 4
q.remove(); // 2 3 4
q.element(); // 2; no removal
q.push(5); // 5 2 3 4
q.pop(); // 2 3 4 // same as remove()
q.peek(); // 2; no removal
q.addFirst(5); // 5 2 3 4
q.addLast(6); // 5 2 3 4 6
q.getFirst(); // 5; no removal
q.getLast(); // 6; no removal
q.removeFirst(); // 2 3 4 6
q.removeLast(); // 2 3 4
Methods are available which throw exceptions when something goes wrong and corresponding methods are available which do the same but don’t throw any exceptions.
q.element(); // throws exception if queue is empty
q.peek(); // never throws exception
q.add(Element); // exception
q.offer(Element); // no exception
q.remove(); // exception
q.poll(); // no exception
If we try and get
an element from an empty collection, it will throw NoSuchElementException
at runtime (unchecked exception).
Key-value pairs and access time is linear. Can only contain duplicate values, not keys.
All are immutable.
// Factory Methods
Map.of("key1", "value1", "key2", "value2");
Map.ofEntries(
Map.entry("key1", "value1"),
Map.entry("key2", "value2"));
Map.copyOf(collection)
They all create immutable maps, but the differences are:
Map.of()
is slightly weird syntax because of the boxing of adjacent values as a KV pair, also it can max have 10 KV pairs, unlike Collection methods like List.of()
, Set.of()
, etc…Map.ofEntries()
introduced in Java 9 allows creation of more than 10 KV pairs, that too in a more intuitive syntax.Map<String, String> map = new HashMap<>();
map.put("A", "Apple");
map.get("A"); // Apple; if no such key exists then "null" is returned (caution!)
map.put("A", "Apricot"); // "Apricot" will replace "Apple" (keys form a set; no duplicates) (caution!)
map.keySet(); // returns Set of keys, can use in forEach loop
map.values();
map.contains(); // compiler-error; method doesn't exist in Map interface (its a Collection interface method)
map.containsKey("A");
map.containsValue("Apple");
map.size(); // 1
map.forEach((k, v) -> System.out.println(k)); // print all keys; takes BiFunction
map.values().forEach(System.out::println); // prints all values
// advance methods
map.getOrDefault("A", "Alpha");
map.putIfAbsent("C", "Cat"); // otherwise ignore
map.replace("B", "Bear"); // replace value for key "B" with "Bear"
map.replaceAll((k, v) -> k+v); // if key and values are numeric; takes BiFunction
HashMap: unordered keys. Implemented internally using a array of linked lists. notes
TreeMap: has ordered keys unlike HashMap
. We’ve to supply a Comparator
when creating a TreeMap
to specify relative ordering of keys. If a comparator isn’t passed in the TreeMap
constructor call, it orders the keys in their natural order (default). Uses Red-Black Tree internally (a type of self-balancing tree).
// ordering of keys in a HashMap isn't always guaranteed
Map<Integer, String> mp = new HashMap<>();
mp.put(1, "X");
mp.put(2, "Y");
mp.put(3, "Z");
for (var e : mp.keySet()) {
System.err.println(e);
}
// Output: 123
-----------------------------------------------------------------------
// ordering of keys in a TreeMap is always guaranteed
Map<Integer, String> mp = new TreeMap<>(Comparator.reverseOrder()); // changed only this
mp.put(1, "X");
mp.put(2, "Y");
mp.put(3, "Z");
for (var e : mp.keySet()) {
System.err.println(e);
}
// Output: 321
Maps don’t implement Iterator
interface and are thus not iterable. We use a for loop to iterate on them:
// iterate on keys
for(var e : mp.keySet()){
System.out.println(e);
}
// iterate on values
for(var e : mp.values()){
System.out.println(e);
}
A map’s keys are unique and form a Set. Hence we can also get entrySet (Set<Entry<K, V>>
) of map and iterate on it:
// iterate on both keys and values as Entry<K, V>
for(var e : mp.entrySet()){
System.out.println(e.getKey() + ", " + e.getValue());
}
// the Set collection is iterable so we can use iterator with it too
Another piece of code helpful in building a frequency map:
String str = "foobar";
Map<Character, Integer> mp = new HashMap<>();
// normal verbose way
for(Character c : str.toCharArray()){
if(mp.containsKey(c)) {
mp.put(c, mp.get(c) + 1);
} else{
mp.put(c, 1);
}
}
// better shorter way
for(Character c : str.toCharArray()){
mp.put(c, mp.getOrDefault(c, 0) + 1);
}
// lesser known short way
for(Character c : str.toCharArray()){
mp.merge(c, 1, Integer::sum);
}
A functional interface from java.lang.Comparable
(no need to import
). Uses compareTo()
method, takes 1 argument and returns an int
.
public class Foo implements Comparable<Foo>{
private String name;
public int compareTo(Foo f){
return name.compareTo(f.name); // asc sort by name; uses String's compareTo()
}
}
List<Foo> fooList = new ArrayList<>();
// ...
Collections.sort(fooList);
negative number (< 0) - current object less than argument object
zero (= 0) - current object equal to argument object
positive number (> 0) - current object greater than argument object
// to compare numeric types
public class Animal implements Comparable<Animal> {
private int id;
public int compareTo(Animal a) {
return id - a.id; // sorts ascending by id
}
public static void main(String[] args) {
var a1 = new Animal();
var a2 = new Animal();
a1.id = 5;
a2.id = 7;
System.out.println(a1.compareTo(a2)); // -2
System.out.println(a1.compareTo(a1)); // 0
System.out.println(a2.compareTo(a1)); // 2
}}
// now if we create a List<Animal>, we can use below statement to sort since compareTo() has been defined for our Animal class
Collections.sort(animalList);
If we don’t specify generic type, the argument is Object o
type. And we can always cast it to proper type inside the method to compare.
public class LegacyWay implements Comparable{ // no generic type specified
public int compareTo(Object obj) {
LegacyWay d = (LegacyWay) obj; // cast because no generics
return name.compareTo(d.name); // assumes name is String type
}}
It is a good programming practice to keep equals()
and compareTo()
consistent with each other.
Also, a functional interface from from java.util.Comparator
. Uses compare()
method, takes 2 arguments and returns int
.
Comparator<Duck> byWeight = new Comparator<Duck>() { // anon class impl
public int compare(Duck d1, Duck d2) { // override and impl
return d1.getWeight() - d2.getWeight();
}};
Collections.sort(ducks, byWeight); // use
Comparator<Duck> byWeight = (d1, d2) -> d1.getWeight() - d2.getWeight(); // using lambda and getter
// same as above but much shorter using direct field access without getter
Collections.sort(list, (d1, d2) -> d1.weight - d2.weight);
Comparator<Duck> byWeight = Comparator.comparing(Duck::getWeight); // another way using Comparator.comparing() static method which generates a lambda automatically
Comparator.comparing() - builds a comparator. Provide getter method for a field to be used for comparison as method reference (keyExtractor), returns a suitable Comparator.
Comparator.comparing(Function); // compare by results of a method that returns any Wrapper class object
Comparator.comparingInt(ToIntFunction); // compare by results of a method that returns a primitive int
Comparator.comparingLong(ToLongFunction); // compare by results of a method that returns a primitive long
Comparator.comparingDouble(ToDoubleFunction); // compare by results of a method that returns a primitive double
Comparator.naturalOrder(); // no need to specify any field/getter; works only on known types like Integer, String, etc..
Comparator.reverseOrder(); // no need to specify any field/getter; works only on known types like Integer, String, etc..
For comparing objects of classes which don’t have a natural ordering or implement Comparable
interface, we need to provide a Comparator for comparing them i.e. Comparator.comparing(Function, Comparator.comparing(Function))
:
// comparing Student objects based on method that returns object of "Foo"
Comparator.comparing(Student::getFoo);
// but "Foo" itself needs comparator to identify ordering, here we used the "rank" field of "Foo" for that
Comparator.comparing(Student::getFoo, Comparator.comparing(Foo::getRank));
Comparator Chaining with available methods:
.reversed();
.thenComparing(Function); // if previous comparator returns 0 (a tie), run this, otherwise return the value from previous
.thenComparingInt(ToIntFunction);
.thenComparingLong(ToLongFunction);
.thenComparingDouble(ToDoubleFunction);
// example
Comparator<Foo> c = Comparator.comparing(Foo::getMarks).thenComparing(Foo::getName);
// compare by marks; or by name if marks comparison is a tie
// two overloaded versions of sort method of Collections class
Collections.sort(List<T> list)
Collections.sort(List<T> list, Comparator<? super T> c)
Collections.sort()
can use both Comparable
as well as Comparator
. Returns void
, modifies the collection supplied to it in-place.
Comparable<T>
is usually implemented within the class whose objects are being compared (T
) as int compareTo(T a)
override, called implicitly when we use Collections.sort(fooList)
.Comparator
’s int compare(U a, U b)
method is usually implemented using lambda expression passed as the second argument e.g. Collections.sort(fooList, fooComp)
.Other ways to sort:
Collections.reverse(collection); // reverses whatever current order is there
// sorting a list; List's sort() method also takes in a Comparator<T>
listObj.sort((d1, d2) -> d1.marks - d2.marks);
// alternatively
listObj.sort(Comparator.comparing(Student::getMarks));
// use null as parameter for natural ordering
listObj.sort(null);
// alternatively; explicitly specify comparator
listObj.sort(Comparator.naturalOrder());
An object used to loop (iterate) through collections (based on Iterator design pattern).
List<Integer> numbers = new ArrayList<>();
numbers.add(12);
numbers.add(8);
numbers.add(2);
numbers.add(23);
Iterator<Integer> it = numbers.iterator(); // alternatively, use var here
while(it.hasNext()) {
Integer i = it.next(); // get an element
if(i < 10) {
it.remove(); // remove numbers with value < 10
}
}
System.out.println(numbers);
What happens if underlying collection is modified after the iterator has been created? Iterator may throw a ConcurrentModificationException
(best effort basis).
Collections maintain an internal counter called modCount
. Each time an item is added or removed from the Collection, this counter gets incremented. When iterating, on each next()
call, the current value of modCount
gets compared with the initial value. If there’s a mismatch, it throws ConcurrentModificationException
which aborts the entire operation.
Iterator<Integer> it = numbers.iterator();
while (it.hasNext()) {
Integer n = it.next();
numbers.add(50); // ConcurrentModificationException
}
Removal is fine if done with Iterator’s it.remove()
method and not Collection Impl class list.remove()
method:
while (it.hasNext()) {
if (it.next() == 40) {
it.remove(); // fine; removes 40
}
}
while (it.hasNext()) {
if (it.next() == 40) {
numbers.remove(0); // ConcurrentModificationException
}
}
Only structural modifications (add/remove elements) leads to a ConcurrentModificationException
with FF iterators; and not updation of existing elements. Ex - update to a list using list.set(2, "X")
.
Iterators on Collections from java.util.concurrent
package such as ConcurrentHashMap
, CopyOnWriteArrayList
, etc.. are Fail-Safe in nature.
They are aware of the elements being added to the underlying collection and update modCount
accordingly concurrently.
ConcurrentHashMap<String, Integer> map = new ConcurrentHashMap<>();
map.put("First", 10);
map.put("Second", 20);
Iterator<String> it = map.keySet().iterator();
while (it.hasNext()) {
String key = it.next();
map.put("Third", 30); // fine; loop executes 3 times
}
Offers more collections such as ImmutableMap
, BiMap
(map which can be accessed via values and keys as well!), MultiMap
(map a key to multiple values!), etc…
Most popular are:
Comparison: https://www.baeldung.com/apache-commons-collections-vs-guava
Guava Demo: https://www.baeldung.com/guava-collections
/java/misc/#top-interview-questions-on-collections-internals