How to execute code before the main()
method? Write the code in static{ }
block of the class containing main()
method so that it runs during class initialization. Doing it in the instance initializer block won’t work since JVM doesn’t call this class (and main()
method) using a class instance.
public class Test {
public static void main(String[] args) {
System.out.print("foo");
}
static { System.out.print("bar"); }
}
// Output: "barfoo"
Integer
wrapper class tricky question on JVM Integer Cache:
Integer a = 127;
Integer b = 127;
System.out.println(a == b); // true; uses cached integer object ref for both
Integer a = 128;
Integer b = 128;
System.out.println(a == b); // false; uses diff refs
/* For Integer objects, Java caches instances of Integer values in the range of -128 to 127. This is known as the "Integer Cache".
When you create an Integer within this range, Java will return a reference to an existing cached object rather than creating a new one. */
Minor trick questions:
// operator association
System.out.println("Asd" + 10 + 20); // "Asd1020"
System.out.println(10 + 20 + "Asd"); // "30Asd"
// string immutability
String str = "Arya";
str.replace("A", "x");
System.out.println(str); // "Arya"
str = str.replace("A", "x");
System.out.println(str); // "xrya"
str.toUpperCase(); // same as above
Floating-point limitations:
System.out.println(0.1 + 0.2 == 0.3); // false; its 0.30000000000000004
// nothing new here, all popular programming languages suffer from the same issue because they all use IEEE-754 (aka Double Precision Floating Point Numbers).
Shallow Copy (reference variables), Deep Copy (copying data to new object manually), Cloning (copying data to new object with clone() method) newObj = oldObj.clone()
Cloning can be customized overriding the clone()
method of the Object
class, and indicating by implementing Cloneable
marker interface.
Marker Interfaces: Interfaces that have no methods and constants defined in them. They are there for helping the compiler and JVM to get run time-related information regarding the objects. Ex - Serializable
, Cloneable
interface.
>>
vs >>>
operator: link
Double-Brace Initialization:
Set<String> countries = new HashSet<String>() {
{
add("India");
add("USSR");
add("USA");
}
};
// define Anonymous Class
// run code inside Instance Initializer Block
String’s length() isn’t accurate - counts code points rather than no. of chars, code points can be more than one for characters like an emoji since its UTF-16. link
3 GC scenarios: reference becomes null, reference points to some other object, Island of Isolation
Island of Isolation: The way Mark & Sweep algorithm work is that it starts the scan from the GC root (main()
method) and follows all references. Everything that is marked (reachable) is not removed, everything else is sweeped (removed).
// normal GC scenario
String s1 = new String("Foo");
s1 = new String("Bar");
// String object with value "Foo" isn't referenced anywhere from GC root; hence its GC'ed later
// Island of Isolation
public class Test {
Test ib;
public static void main(String [] str){
Test t1 = new Test();
Test t2 = new Test();
t1.ib = t2; // t1 points to t2
t2.ib = t1; // t2 points to t1
t1 = null;
t2 = null;
/*
* t1 and t2 objects refer to each other
* but have no references from GC root
* these 2 objects form an Island of Isolation and are eligible for GC
*/
}
}
wait()
and notify()
mechanism in Thread: call wait() on a object in multiple threads to make them all wait, we can then call notify() from another class to wake up any one random thread waiting, use notifyAll()
to wake up all waiting threads. Mandatory condition is that the object on which we are synchronizing must be the same object on which we’re calling the wait and notify methods. reference
// arbitrary object used as a monitor
Mon object = new Mon();
// in waiting thread
synchronized(object){
object.wait();
// releases lock on "object" and waits indefinitely
}
// in notifying thread
synchronized(object){
object.notify();
// notifies any random thread waiting on "object" but doesn't release lock instantly, only after this sync block finishes
}
We wrap these in sync block because monitor (object
) needs to be available for locking before we do wait
or notify
on it. Be very careful when designing to avoid deadlocks and other concurrency issues.
wait()
vs sleep()
: sleep makes thread TIMED_WAIT
that gets over after timeout (thread doesn’t release lock), wait
makes thread lose its ownership (releases lock), state as BLOCKED
and must be notified from outside using notify
or notifyAll
method using the same object as a monitor for sync block.
String being immutable is a:
char[]
to store passwords and manual erasure of each element is possible (as opposed to String as they are immutable) as soon as its work is done.Objects
utility class has static methods to check not null etc… Ex - Objects.nonNull(myObj)
.
Diamond Operator: <>
optional to provide any type information, still need to be used as a placeholder. Ex - List<String> list = new ArrayList<>()
Utility Class: Java provides many utility classes that are final
, can’t be instantiated (private
constructor), and contains only static
methods that often take in an instance. Ex - Files
, Executors
, Objects
, etc…
Iterator vs ListIterator: ListIterator
is a more specific interface of the Iterator
interface i.e. public interface ListIterator<E> extends Iterator<E>
. It can traverse in forward (it.next()
) as well as backward direction (it.previous()
) and can also add or update elements of the underlying data structure unlike the generic Iterator
.
List<Integer> list = new ArrayList<>();
ListIterator<Integer> it = list.listIterator();
it.next();
it.previous();
it.add(1);
it.set(2);
They are only available to use for List
interface and its impl classes like ArrayList
, LinkedList
, Vector
, and Stack
(last two are legacy).
Fail-fast vs Fail-safe Iterators: Iterator
on FF on classic collections, and on thread-safe collections (CopyOnWriteList
, ConcurrentHashMap
, etc) they are FS. notes link
What is Load Factor? It is the threshold at which a data structure (collection) is resized to accomodate for future incoming data. Default is 0.75
which means that when 75% of the current size is populated, the collection will be resized to a bigger one.
Resizing after Load Factor is reached: Memory locations are contiguous in collections like List and HashMap so data has to be “copied over” to the newly allocated memory location (expensive). We can’t simple “stretch” and resize as we may have memory being occupied by other programs surrounding the original allocated memory. In case of HashMap this is even more expensive as a rehash is required (since backing array size changes; see below notes section on HashMap).
Can null
element be added to a collection? only one null
can be added to HashMap
and HashSet
. For all the other non-major collections, it depends. Ex - it can’t be added to TreeSet
(as a Comparator
is required here) but allowed in LinkedHashSet
.
Collection vs List remove()
method is overloaded:
// List and ArrayList have two remove methods overloaded
List<Integer> list = new ArrayList<>(List.of(1, 2, 3));
list.remove(2); // calls remove(int index); closest without autoboxing
System.out.println(list);
// Output: [1, 2]
Collection<Integer> list = new ArrayList<>(List.of(1, 2, 3));
list.remove(2); // calls remove(Object o) of Collection interface
System.out.println(list);
// Output: [1, 3]
The interface Collection
doesn’t have a remove(int index)
method because the referred collection can be unindexed like a Set
, so it just has the remove(Object o)
method available at its level of abstraction.
var keyword tricky questions:
var list1 = new ArrayList<>(); // ArrayList<Object>
var list2 = new ArrayList<String>(); // ArrayList<String>
var list3 = List.of("foo", "bar"); // List<String> (Java infers it automatically!)
Intersection Type: rather than using a common superclass ref, we can also specify a type using Generics as follows:
// a generic method using a superclass ref
public void process(FlyableAndSwimmable animal) {
animal.fly();
animal.swim();
}
// same generic method with a intersection type
public <T extends Flyable & Swimmable> void process(T animal) {
animal.fly();
animal.swim();
}
// Benefit: saves the effort for creating a new interface just to get a new combined type to use.
The JVM uses this for implicit typing too if we use var
keyword:
var list = List.of(2, "foo"); // List<Serializable & Comparable<...> & Constable & ConstantDesc>
// note that JVM could've taken the type as List<Object> but it didn't and used an intersection type instead.
We can’t explicitly create intersection types in normal code but can only use them if we use Java Generics (in method params or as a return type).
equals() and hashCode() contract: Two objects that return true for equals()
must also have the same hashCode()
value. Hence if we override one, we must override the other too.
Typically, if they are not overriden with custom impl, equals()
calc based on the memory address of the instance in the heap, and hashCode()
calc based on the memory address too. We often need to override and provide custom impl for these methods based entirely on instance members and not its memory address.
In the below code, identical objects are considered diff incorrectly since default hashCode()
isn’t overriden and its inherited from java.lang.Object
class:
Set<Student> set = new HashSet<>();
set.add(new Student(1, "A"));
set.add(new Student(1, "A"));
System.out.println(set.size()); // 2
Override default impl of the hashCode()
method of Student
class based on its contents.
// Ex - overriding for Student class
@Override
public boolean equals(Object obj){
if(this == obj){
return true;
}
Student studentObj = (Student) obj; // explicit cast (Object to Student)
if(this.id == studentObj.id && this.name.equals(studentObj.name)){
return true;
}
return false;
}
@Override
public int hashCode(){
return Objects.hash(this.id, this.name); // utility class; varargs and handles null implicitly
}
Objects.hash()
utility class static method uses a variant of the well-known hash function called “MurmurHash” and produces variable length integral numbers as output hash.
Do note that they are properly overriden on Wrapper classes as well as core classes like String
(compare contents rather than memory address).
Set<String> st = new HashSet<>();
st.add("ABC");
st.add("ABC");
System.out.println(st.size()); // 1
Underlying data structures are backing array of size 16
(buckets) and a linked list attached to each bucket cell.
Equivalent of unordered_map<>
in C++ which also uses hash table (array/buckets) internally.
Methods to manage elements are hashCode()
and equals()
that are present in every class (inherited from java.lang.Object
class) and need to be overwritten for proper functioning of Map.
HashMap
class’s put(k, v)
method:
V put(K key, V value){
int hash = hash(key); // calc hash
putVal(hash, key, value); // puts data in linked list
}
int hash(K key){
key.hashCode() % 16; // same but using bitwise operation
}
A node in the attached backing linked lists of HashMap
:
┌────────────────┬──────────────────┬─────────────────┬─────────────────┐
│ key │ value │ hash │ next │
└────────────────┴──────────────────┴─────────────────┴─────────────────┘
put() operation: There are bound to be hash collisions since the backing array is only of size 16
, whenever there is a non-zero sized LL present on the bucket we are hashed to, we check (linear search) if the key already exists by comparing hash
value present in LL nodes and then verify again by comparing the key we’re currently searching and key
value from LL using equals()
method and replace if they are equal otherwise attach new node to the end of linked list at that bucket.
There can only be one null
key in a HashMap and it is written to bucket indexed as 0
.
Map<Integer, String> mp = new HashMap<>();
mp.put(null, "ABC");
System.out.println(mp.get(null)); // prints "ABC"
get() operation: we goto bucket and check for non-zero sized LL, if nodes are present linearly search them for hash
value match and once found verify with using equals()
on search key and LL nodes key.
Worst case TC for get()
and put()
operations = O(n)
, considering every key hashes to the same bucket (unequally distributed hash function).
Performance improvement of HashMap (since Java 8): buckets are limited (only 16), collisions are more likely to occur with increasing map entries leading to poor performance. When a bucket’s LL grows beyond a certain threshold, it switches from using LL (O(n)
operations) to using a Balanced Tree (specifically, a Red-Black Tree) to maintain performance.
Note that TreeMap
(ordered Maps) also use Red-Black Trees internally in Java.
A HashMap’s initial default capacity is 16, default load factor is 0.75. So what happens when load factor is bypassed?
Just like lists and other collections, HashMap’s capacity has to be increased when load factor is crossed. But since HashMap use hashing and it is dependent upon the number of buckets (N
) in the backing array. We need to rehash the keys and store them in the newly allocated memory locations, a relatively more expensive operation than Lists.
It uses HashMap
internally! Keys are the entities we want to put in the set and value is constant PRESENT
(random instance of Object
class) if the entity exists in the set.
HashSet
class:
static final Object PRESENT = new Object();
public HashSet() {
map = new HashMap<>();
}
public boolean add(E e) {
return map.put(e, PRESENT) == null;
}
public boolean remove(Object o) {
return map.remove(o) == PRESENT;
}
Immutability of Keys: Keys are nothing but instances but they must be immutable if they’re being put into a set. This is to avoid miss (unable to find key) for instances that were added to the set previously and were modifed after the addition.
As a solution, we can add the modified instance again after modification and it will also be put into the set. But beware that if the instance has a timestamp
member then we end up having a lot of copies of the same instance (redundancy) differing only by timestamp.
It is just like a normal HashMap
but attaches nodes across buckets in a circular doubly-linked list fashion using before
and after
links in LL node so that it can maintain ordering. It is not thread-safe.
There is a dummy Head
node which is present in the impl of LinkedHashMap
to ensure addition of Circular DLL nodes relative to it. A newly inserted or accessed node is inserted at the end of the list (before
link of Head
node). Conversely, the least recently used (LRU) element is at the beginning of the list, just after
the Head
node. This way it follows the fashion ... -> 1 -> 2 -> 3 -> Head -> 1 -> ...
Ordering is preserved either by - insertion order (default) or access order. Used to implement LRU cache when access ordering is enabled.
public class LinkedHashMap extends HashMap implements Map { }
// constructors
new LinkedHashMap(); // default; capacity = 16, LF = 0.75, order = insertion
new LinkedHashMap(int capacity); // LF = 0.75
new LinkedHashMap(int capacity, float loadFactor, boolean accessOrder); // true = accessOrder; false = insertion order
If insertion order is followed, access (get()
) won’t lead to any structural modifications. But, on every access in a access ordered map, the links to nodes rearranges (to maintain LRU item’s constant time access).
A node in the attached backing linked lists of LinkedHashMap
:
┌──────────────┬──────────────┬──────────────┬─────────────┬──────────────┬──────────────┐
│ key │ value │ hash │ next │ before │ after │
└──────────────┴──────────────┴──────────────┴─────────────┴──────────────┴──────────────┘
The answer here makes the relatively low-complexity of LinkedHashMap
over HashMap
more clear - link.
Performance overhead is more than a normal HashMap due to links rearranging and a slightly more memory footprint because of the bigger DLL nodes.
There is a LinkedHashSet
too which uses LinkedHashMap
as its underlying data struture, just like HashMap
and HashSet
.
Part of java.util.concurrent
package.
Use this when thread-safety is required and changes to the underlying data structure shouldn’t throw ConcurrentModificationException
.
Features:
In a Collections.syncronizedMap(mp)
, whole map is locked even for reads! It wraps all methods and code in sort-of synchronized
method/block, but only a single thread can access the code at a given time, no matter which code - read or write.
It doesn’t use equals()
or hashCode()
for its operations. Rather it uses reference equality operator (==
) and the JVM provided identity hashcode.
Since it doesn’t use the equals()
and hashCode()
methods at all, overriding them is optional and hence there is no gurantee that the hashCode()
and equals()
contract is satisfied for objects being put as key in an IdentityHashMap
.
It doesn’t face the mutable key problem as a key object’s instance members can be modified after adding the object to the Map and it will still have the same memory address. Unlike equals()
which changes based on object members.
IdentityHashMap
isn’t thread-safe natively but we can always use Collections synchronizedMap(identityHashMap)
to make it sync.
Map<String, String> identityHashMap = new IdentityHashMap<>();
identityHashMap.put(new String("foo"), "John");
identityHashMap.put(new String("bar"), "Doe");
identityHashMap.put(new String("foo"), "Alice"); // adding duplicate key
System.out.println(identityHashMap.size()); // 3; would've been 2 if any other HashMap type
Usage: it is used when we want to cache all objects (base uniqueness on their object reference), takes up a lot of storage but keeps record of all unique objects created in the system.
No segment locking unlike ConcurrentHashMap
but employs a diff strategy to achieve thread-safety.
A copy of the list is created and modified on writes, it is then used to replace the original copy so that changes get reflected in the original. On a read, just read from the original as no locking is required as no concurrent modification is taking place on the original copy.
This has significant processing overhead and memory footprint compared to normal List implementations.
Java 21 was GA released on Sept, 2023. It is the latest LTS release.
Next LTS release will be Java 25 in Sept, 2025 (Oracle’s 2 year cycle for LTS releases).
switch
:
Pattern matching in Java refers to a feature that allows you to check an object against a given type and, if the object is of that type, extract a variable of that type in a single step. Basically instanceof
operator usage.Conventional way is by using instanceof
operator with if-else
ladder but since Java 21 we can use it in switch
block/expression.
// Pattern matching using instanceof
// object "o" is checked if its of type String
// if it is, then its put in a new reference variable "s" and we can call "length()" method in it
// otherwise calling "length()" method on object "o" would be a compiler error
Object o = "foobar";
if (o instanceof String s){
System.out.println("This is a String of length " + s.length());
} else {
System.out.println("This is not a String");
}
// Pattern matching using switch expression (new feature)
Object o = ...; // any object
String formatter = switch(o) {
case Integer i -> String.format("int %d", i);
case Long l -> String.format("long %d", l);
case Double d -> String.format("double %f", d);
case Object o -> String.format("Object %s", o.toString());
}
record Point(int x, int y) {}
Object o = ...; // any object
if (o instanceof Point(int x, int y)) {
// do something with x and y
}
Reference: https://dev.java/learn/pattern-matching/
Java 22 was GA released on March, 2024. It is the release for the first half of 2024.
We often use the variable _
in Go to ignore index while iterating using a for range loop:
msg := "foobar"
// iterate over the string using a for loop and ignore the index
for _, c := range msg {
fmt.Printf("%c\n", c)
}
We can now do something similar to the above in Java 22 onwards in for-each loop:
for (var _ : nums) {
cnt++;
if (cnt > limit) {
// don't use _ loop variable in business logic here
}
}
this()
(call to another overloaded constructor) or super()
(call to superclass constructor).This is changed in Java 22! It now allows statements before the super()
(or this()
) constructor call.
public Test(){
System.out.println("foobar"); // valid
super();
}
Regional Pinning for G1GC: It is a performance optimization technique for the G1 Garbage Collector (G1GC). G1GC partitions the heap into fixed-size memory region. During garbage collection cycles, it identifies and collects regions with a high concentration of dead objects, improving memory efficiency.
Region pinning allows developers to designate specific memory regions as pinned, preventing them from being moved during garbage collection cycles.
This is super useful for frequently accessed data. Pinning those regions ensures the data remains readily available in its current location, potentially reducing memory access times and improving application performance.
@
-tags.Lots of preview features like better constructors and pattern matching with switch
were continued as preview features. And String Templates was dropped completely.