Kotlin
Kotlin
Table of contents
Topics
Advantages Over Java
Data Class
Extension Functions
KMM
Concise Code
Support for Null Safety
: Kotlin provides Safe Call (?.), Elvis (?:) and Not Null Assertion (!!) operators which define what needs to be done in case of a null encounter.- Safe call (?.): Safe calls allow you to safely access properties and methods on nullable objects without risking an NPE. If the object is null, the call returns null instead of throwing an exception.
name?.toLowerCase() OR if(name!=null) name.toLowerCase() else null
- Elvis Operator(?:): Operator used to return a non-null or default value. It returns the left value if it’s not null, otherwise returns the right value.
val a = b ?: "defaultValue" OR val a = if (b!=null) b else "defaultValue"
- Not null assertion Operator(!!): promising that the value will not be null, but if found null, will cause null pointer exception
- Safe call (?.): Safe calls allow you to safely access properties and methods on nullable objects without risking an NPE. If the object is null, the call returns null instead of throwing an exception.
Basics
?:
: Elvis Operator, for safe call, unwraps the value from Nullable!!
: value will not be null, if null will cause runtime crash==
: check if values are equal or not===
: check if reference is equal or notinit
: it is the initialiser block in Kotlin, it’s executed once the primary constructor is instantiated.
Keywords
var
: variable value can be changedval
: variable value cannot be changed, sets at runtimecont val
: makes val compile time constant, hence must be known at compile timeopen
: make the class open for inheritance by other classes as classes are final by default in kotlin.when
: substitute for the switch case of javawhen(a){ in 1..10 -> println("Value Less Than 10") in 11..100 -> println("Value More Than 10") }
- ⭐
lazy
: is as method or rather say lamba exppression, it’s set on val only, the val would be created at runtime only when it’s required, should be used for time-consuming intializations, so that code can be faster and efficient. - ⭐
lateinit
: used to declare the variable with a gurantee to initialize it before using, otherwise would throw exception, used with var. UseisInitialized
to check if variable has been initialized. - ⭐
object
: used to create Singleton, ensure that only one instance of that class is created even if 2 threads try to create it, it’s a lazy instance, hence will be created once the object is accessed, otherwise it won’t even be created. - ⭐
Q: Are
Object
declarations thread safe, even if two threads tries to create it at same time? A: Yes, object is thread safe by construction. As it’s just final class with static instance initializations, when decompiled. ref companion object
: similar to static method in java, to access something by their class name without having the instance of class.
Visiblity Modifiers
public
: default visibility modifier, are accessible everywhere, both, within and outside the module.internal
: visibile only within the same moduleprotected
: allows a declaration to be accessible within the same class and its subclasses.private
: only visible within the scope where they are declared.
Classes
nested class
: Class inside another class, but by default nested class don’t have access to members of the outer class.inner class
: makes the members of outer class accessible by the nested class by adding the inner keyword to nested class.- ⭐
data classes
: used to store values, data. Generates equals(), hashCode(), copy(), toString() automatically, which can be overriden.data class User(val name, val string)
cons:
- Can’t be inherited, but can inherit.
- a data class may have properties that are not part of its constructor, they will not be regarded in any of the functions that are generated.
data class DollarBill(val amount: Int) { var serialNumber: String? = null }
enum classes
: used to model types that represent a finite set of distinct valuesenum class SchnauzerBreed(val height: Int) { MINIATURE(33), STANDARD(47), GIANT(65); val family: String = "Schnauzer" fun isShorterThan(centimeters: Int) = height < centimeters } println(SchnauzerBreed.STANDARD.family) // Prints "Schnauzer" println(SchnauzerBreed.STANDARD.isShorterThan(40)) // Prints "false"
- Built-in properties offered by enum:
- ordinal: position of item in list
- name:
- Built-in properties offered by enum:
abstract classes
: Abstract classes can be extended by other classes. Their functions and properties can be:abstract
, in which case they have no body in the abstract class, but subclasses must implement them.open
, in which case they have a body in the abstract class, but subclasses may override them.Final
(i.e., neither abstract nor open), in which case subclasses cannot override them.
open classes
: An open class is a class that can be both extended and instantiated directly. Open classes cannot contain abstract members.sealed classes
: A sealed class is, by definition, also an abstract class. This means that you can’t directly instantiate it - you can only instantiate one of its subclasses. Hence, if you want exhaustive subtype matching, you’ll need to include the sealed modifier. If you need to limit values, then use an enum class. If you need to limit types, then use a sealed type.
OR
A sealed class is a class that restricts which other classes can inherit from it. All subclasses must be declared within the same file as the sealed class itself.sealed class Shape { class Circle(val radius: Double) : Shape() class Rectangle(val width: Double, val height: Double) : Shape() object NotAShape : Shape() }
- Benefits of sealed class:
Restrictive Inheritance
: All subclasses of a sealed class are known at compile-time.Exhaustive when Expressions
: When usingwhen
expressions with sealed classes, Kotlin ensures that all possible subclasses are handled, leading to safer and more maintainable code. ref
Reference Equality
reference equality
: when two variables refer to same object instance, then they will be considered equal.value equality
: when objects are considered equal based on their property values rather than their identityreferential equality operator
: Even when you override the equals() function to give a class value equality, you can still check to see whether two variables refer to the same instance. ```kotlin //reference equality class DollarBill(val amount: Int) val bill1 = DollarBill(5) val bill2 = DollarBill(5) println(bill1 == bill2) // false
//value equality class DollarBill(val amount: Int) { override fun equals(other: Any?) = if (other is DollarBill) amount.equals(other.amount) else false } println(bill1 == bill2) // true
//referential equality operator println(bill1 == bill2) // false
### <a name='Constructors'></a>Constructors
[🔝](#table-of-contents)
- `Primary Constructor`: The primary constructor is defined in the class header, directly after the class name, and is part of the class declaration itself.
- Can use var and val
- no use of constructor keyword
- implicityly called when an instance of class is created
```kotlin
class User(val name: String, val age: Int){
init {
}
}
Secondary Constructor
: Also knows an auxiliary constructor, is an additional constructor you can define within the class body using the constructor keyword. Allows to provide alternate ways to instantiate an object.- can call the primary constructor using this()
- each constructor can delegate to primary constructor or another secondary constructor.
class User(val name: String, val age: Int) { constructor(name: String) : this(name, 22) { println("User created with name: $name, age: $age") } }
Init Block
: The init block is used for initializing properties or executing startup logic right after an object is created. It runs after the primary constructor and before any secondary constructor.- called automatically when an instance of class is created
- can have multiple init blocks and they will be executed the same order they are defined in the class.
- any property defined in primary constructor can be accessed and used inside the init block.
class User(val name: String, val age: Int) { init { println("name: $name and age: $age") } }
Functions
Higher Order Functions
: Functions the either takes function/LE as parameter, or returns a function/LE
//this is hof because it takes function as parameter
fun calculateTotal(initialPrice: Double, applyDiscount: (Double) -> Double): Double {
val priceAfterDiscount = applyDiscount(initialPrice)
val total = priceAfterDiscount * taxMultiplier
return total
}
//this is hof because it returns a function
fun discountForCouponCode(couponCode: String): (Double) -> Double =
when(couponCode) {
"FIVE_BUCKS" -> ::discountFiveDollars
"TAKE_TEN" -> ::discountTenPercent
else -> ::noDiscount
}
Lambda
: Lambda expression is nothing but simplified representation of function aka function literals, which can be passed as parameter, stored in a variable, even returned as a value.
val applyDiscount: (Double, Double) -> Double = { price: Double, discount: Double -> price - discount }
// opening braces - paramters - arrow - function body - closing braces
//Lambdas with multiple statements
//In Kotlin, writing the lambda outside of the parentheses like this is called trailing lambda syntax.
val withFiveDollarsOff = calculateTotal(20.0) { price ->
val result = price - 5.0
println("Discounted price: $result")
result //result of function, without return keyword
}
Anonymous Function
: Because lambdas have no name, many languages regard lambdas as Anonymous Function but in Kotlin the term anonymous function refers to another way of writing the function literal.val applyDiscount: (Double) -> Double = fun(price: Double): Double { return price - 5.0 }
- ⭐
Extension Functions
: When you pick up a receiver class, and dynamically inject a member function into the class which acts pretty much same as regulary defined member functions of that class.
fun View.hide() {
this.visibility = View.INVISIBLE
}
appbar.hide()
Infix Function
:Inline Function
: When a function is made inline function, compiler will treat the function code as direct code in the calling function instead of seperate function.
fun main(){
printMessage()
}
inline fun printMessage(){
println("Hehe")
}
Becomes ->
fun main(){
println("Hehe")
}
Scope Functions
Scope
: Section of code where you can declare or use any particular varibale or function.Statement Scope
: You can only use something that was declared in a statement scope after the point where it was declared.fun circumference(): Double { fun diameter() = radius * 2 //right val result = pi * diameter() fun diameter() = radius * 2 //wrong return result }
Declaration Scope
: Unlike statement scopes, things declared within a declaration scope can be used from a point in the code either before or after that declaration.class Circle(val radius: Double) { val circumference = pi * diameter() fun diameter() = radius * 2 }
Scope Functions
: Functions whose sole purpose is to execute a block of code within the context of an object. When called with a lambda expression provided, a temporary scope is formed, inside which we can access the object without its name. The point of a scope function is to take an existing object - called a context object - and represent it in a particular way inside that new scope.with
: introduces a new scope (the lambda) in which the context object is represented as an implicit receiver (an implicit receiver can be used with no variable name at all)//before address.street1 = "9801 Maple Ave" address.street2 = "Apartment 255" //after with(address){ street1 = "9801 Maple Ave" street2 = "Apartment 255" }
run
: works same awith
but it’s an extension function instead of a normal, top-level function. As it’s a extension function, it can be inserted into a call chain. The run() function returns the result of the lambda.address.run { street1 = "9801 Maple Ave" street2 = "Apartment 255" } val title = "The Robots from Planet X3" val newTitle = title .removePrefix("The ") .run { "'$this'" } .uppercase()
let
: very similar torun()
, but instead of representing the context object as an implicit receiver, it’s represented as the parameter of its lambda.val newTitle = title .removePrefix("The ") .let { it -> "'$it'" } .uppercase()
also
: As with let(), the also() function represents the context object as the lambda parameter, too. However, unlike let(), which returns the result of the lambda, the also() function returns the context object. This makes it a great choice for inserting into a call chain when you want to do something “on the side” - that is, without changing the value at that point in the chain.val newTitle = title .removePrefix("The ") .also { println(it) } .uppercase()
apply
: Like also(), the apply() function returns the context object rather than the result of the lambda. However, like run(), the apply() function represents the context object as the implicit receiver.val newTitle = title .removePrefix("The ") .apply { println(this) } .uppercase()
| |Context|Return |Use Case| |— |— |— |— | |
with
|this |lambda |Grouping Function calls on an object| |run
|this |lambda |Object Configuration and computing the result| |let
|it |lambda |Null Safety Calls| |also
|it |context|Addtional Effects| |apply
|this |context|Object Configuration|
- Scope functions and Null Check: Other than with(), all other scope functions are extension functions, hence we use safe-call operator when calling them, so that they are not called when the receiver is not null.
if (payment != null) { orderCoffee(payment) } payment?.let { orderCoffee(it) } payment?.let { orderCoffee(it) } ?: println("value null")
Questions
- How to concatenate two strings in Kotlin?
- Using String Interpolation:
- Using + or plus() operator:
- Using StringBuilder:
val a = "dsa" val b = "algorithm" val si = "$a $b" val p1 = a+b val p2 = a.plus(b) val sb = StringBuilder().append(a).append(b).toString()
DSA
Tips
var n = (Math.log10(num)) //find number of digits in number
nums[j] = nums[i].also { nums[i] = nums[j] } //swap two numbers in array
var max = maxOf(a,b) //get max value
var min = minOf(a,b) //get min value
Find number of digits in number
:
String
var str = "KOTLIN"
str.size //5
//iterate string
for(char in str)
Arrays
Array<Int>
is an Integer[], means it will be boxed with Integer.parseInt()IntArray
is an int[], in it no boxing will occur, becuase it translates to Java primitive array.val arr = intArrayOf(1,2,3,4,5) var arr = IntArray(10) var arr = IntArray(5) {0} // {0,0,0,0,0} arr.size //4
Matrix
//declare two dimensional array
var matrix = Array(2) { Array<Int>(2) {0} }
println(matrix.contentDeepToString()) //[[0,0],[0,0]]
//declare three dimensional array
var matrix = Array(3) { Array(3) { Array<Int>(3) {0} }}
println(matrix.contentDeepToString())
: [[[0, 0, 0], [0, 0, 0], [0, 0, 0]], [[0, 0, 0], [0, 0, 0], [0, 0, 0]], [[0, 0, 0], [0, 0, 0], [0, 0, 0]]]
List
val list = listOf(1,2,3,4,5)
val list = mutableListOf<Int>() //create empty mutable list
val list = mutableListOf(1,2,3,4,5,6)
val list = List(n) {0} //[0,0,0,0,0]
val list: MutableList<Int> = MutableList(n) {0} //[0,0,0,0,0]
list.sort() //sorts the current list
list.sortDescending()
val sorted = list.sorted() //returns sorted list
val descSorted = list.sortedByDescending()
val even = list.filter {it%2==0} //list with filtered items
val sqaure = list.map {it*it} //transform list and return new list
HashSet
var hashSet = hashSetOf<Int>()
var hashSet = hashSetOf<Int>(1,2,3,4,5,6)
var hashSet: HashSet<String> = hashSetOf<String>()
val mySet = setOf(6,4,29)
// putting into hashSet
hashSet.add(item)
hashSet.addAll(mySet)
// checking
hashSet.contains(item)
hashSet.containsAll(mySet)
// removing
hashSet.remove(item)
hashSet.removeAll(mySet)
// traverssing hashSet
for(item in hashSet){
}
// misc
hashSet.isEmpty()
hashSet.isNotEmpty()
hashSet.clear()
hashSet.size
HashMap
var hashMap: HashMap<String,Int>= HashMap<String,Int>()
for(num in nums) {
frequencyMap[num] = frequencyMap.getOrDefault(num, 0) + 1
}
// printing hashmap
print(hashMap)
// putting into hashmap
hashMap.put("IronMan", 3000)
hashmap["IronMan"] = 3000
hashMap.replace("IronMan",99) // for updating
// getting from hashmap
hashMap.get(key)
// traversing hashmap keys
for(key in hashmap.keys){}
for ((num, freq) in frequencyMap) {}
// misc
hashMap.size
hashMap.isEmpty()
hashMap.clear()
hashMap.containsKey(key)
hashMap.containsValue(value)
hashMap.remove(key)
Queue & Stack
//initialize
val queue: Queue<Int> = LinkedList()
//add
queue.add(n) //if fails, throws exception
queue.offer(n) //returns true if success, otherwise false due to capacity restrictions
//remove
queue.remove() //if failes, throws exception
queue.poll() //returns null if empty
queue.peek()
//misc
queue.size()
queue.clear()
queue.contains(n)
queue.isEmpty()
val stack = Stack<Int>()
stack.push()
x.pop()
x.peek()
Array Deque
// Empty ArrayDeque
val deque = ArrayDeque<Int>()
val dequeWithElements = ArrayDeque(listOf(1, 2, 3))
val isEmpty = deque.isEmpty()
val size = deque.size
val containsElement = deque.contains(10)
val list = deque.toList()
//add
deque.add(10)
deque.addFirst(5)
deque.addLast(20)
//remove
val removedFirst = deque.removeFirst()
val removedLast = deque.removeLast()
//remove specific element (returns true if removed)
val isRemoved = deque.remove(10)
//clear
deque.clear()
//acess
val first = deque.firstOrNull() //peek without removing
val last = deque.lastOrNull() //peek without removing
// Access using indices
val elementAtIndex = deque.elementAt(0)
//iteration
for (element in deque) {
println(element)
}
for (element in deque.asReversed()) {
println(element)
}
Priority Queue
import java.util.PriorityQueue
// Initialization
val pq = PriorityQueue<Int>()
val pqWithElements = PriorityQueue(listOf(5, 10, 1, 3))
val minHeap = PriorityQueue<Int>()
val maxHeap = PriorityQueue<Int>(compareByDescending { it }) //max-heap using custom comparator
val minHeap = PriorityQueue<Pair<Int,Int>>(compareBy {it.second})
// Check if empty
val isEmpty = pq.isEmpty()
val size = pq.size
val list = pq.toList()
// Check if an element exists
val containsElement = pq.contains(10)
// Add elements
pq.add(10)
pq.offer(5) // Alternative to add()
// Remove elements
val removedElement = pq.poll() // Retrieves and removes the highest priority element
val peekedElement = pq.peek() // Retrieves but does not remove the highest priority element
// Clear the PriorityQueue
pq.clear()
// Iterate over elements (unordered because PriorityQueue doesn't maintain insertion order)
for (element in pq) {
println(element)
}
// Example with custom comparator (Max-Heap)
val maxHeapWithElements = PriorityQueue<Int>(compareByDescending { it }).apply {
add(10)
add(1)
add(20)
}
while (maxHeapWithElements.isNotEmpty()) {
println(maxHeapWithElements.poll()) // Elements will be removed in descending order
}
Loop
- val nums = {1,2,3,4,5}
- for(num in nums) l̥
: 1,2,3,4,5
- for(i in nums.indices)
: 0,1,2,3,4
- for(i in 1..5)
: 1,2,3,4,5
- for(i in 5 downTo 1)
: 5,4,3,2,1
- for(i in 1..5 step 2)
: 1,3,5
- val text = "kotlin"
- for(letter in text)
: k,o,t,l,i,n
- for(i in text.indices)
: 0,1,2,3,4
BinaryTree
- Where is it used
- File System
- Databases
- Algo/Networking
- Maths
- Decision Tree: ML
- Compression of files
- Future Data Structures: Head, Graphs
- Types
- Complete Binary Tree:
- All level full
- Last level full from left to right
- Full/Strict Binary Tree
- Each node has either 0 child or 2 child
- Perfect Binary Tree:
- All levels are full
- Height Balanced Tree:
- Avg Height O(logN)
- Skewed Binary Tree:
- Every node has only 1 child
- Ordered Binary Tree:
- Every node has some property or condition
- Like Binary Search Tree
- Complete Binary Tree:
- Algorithms:
- BFS: (When the result lies near the root, need level order calculation)
- Level Order Traversal
- BFS: (When the result lies near the root, need level order calculation)