Kotlin is immensely practical. It addresses the problems that developers have, and not some guys in the academia. So, it has type inference, it has amazing type safety, good collection library, and concurrency library to top it. And it's now official - a lot of organisations are migrating their backend applications to Kotlin, and this trend is not likely to end soon. Follow along to check the most complete and comprehensive collection of the most common and advanced Kotlin Interview Questions every Android developer should know in 2020.
Originally published on FullStack.Cafe - Real Tech Interview Questions And Answers For Devs
Q1: How to initialize an array in Kotlin with values?
Topic: Kotlin
Difficulty: ⭐⭐
In Java an array can be initialized such as:
int numbers[] = new int[] {10, 20, 30, 40, 50}
How does Kotlin's array initialization look like?
val numbers: IntArray = intArrayOf(10, 20, 30, 40, 50)
🔗 Source: stackoverflow.com
Q2: What is basic difference between fold and reduce in Kotlin? When to use which?
Topic: Kotlin
Difficulty: ⭐⭐
-
fold
takes an initial value, and the first invocation of the lambda you pass to it will receive that initial value and the first element of the collection as parameters.
listOf(1, 2, 3).fold(0) { sum, element -> sum + element }
The first call to the lambda will be with parameters 0
and 1
.
Having the ability to pass in an initial value is useful if you have to provide some sort of default value or parameter for your operation.
-
reduce
doesn't take an initial value, but instead starts with the first element of the collection as the accumulator (calledsum
in the following example)
listOf(1, 2, 3).reduce { sum, element -> sum + element }
The first call to the lambda here will be with parameters 1
and 2
.
🔗 Source: stackoverflow.com
Q3: What is the difference between var and val in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐
var is like
general
variable and it's known as a mutable variable in kotlin and can be assigned multiple times.val is like
Final
variable and it's known as immutable in Kotlin and can be initialized only single time.
+----------------+-----------------------------+---------------------------+
| | val | var |
+----------------+-----------------------------+---------------------------+
| Reference type | Immutable(once initialized | Mutable(can able to change|
| | can't be reassigned) | value) |
+----------------+-----------------------------+---------------------------+
| Example | val n = 20 | var n = 20 |
+----------------+-----------------------------+---------------------------+
| In Java | final int n = 20; | int n = 20; |
+----------------+-----------------------------+---------------------------+
🔗 Source: stackoverflow.com
Q4: Where should I use var and where val?
Topic: Kotlin
Difficulty: ⭐⭐
Use var where value is changing frequently. For example while getting location of android device:
var integerVariable : Int? = null
Use val where there is no change in value in whole class. For example you want set textview or button's text programmatically.
val stringVariables : String = "Button's Constant or final Text"
🔗 Source: stackoverflow.com
Q5: What is a data class in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐
We frequently create classes whose main purpose is to hold data. In Kotlin, this is called a data class and is marked as data
:
data class User(val name: String, val age: Int)
To ensure consistency and meaningful behavior of the generated code, data classes have to fulfill the following requirements:
- The primary constructor needs to have at least one parameter;
- All primary constructor parameters need to be marked as val or var;
- Data classes cannot be abstract, open, sealed or inner;
🔗 Source: kotlinlang.org
Q6: How to create singleton in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐
Just use object
.
object SomeSingleton
The above Kotlin object will be compiled to the following equivalent Java code:
public final class SomeSingleton {
public static final SomeSingleton INSTANCE;
private SomeSingleton() {
INSTANCE = (SomeSingleton)this;
System.out.println("init complete");
}
static {
new SomeSingleton();
}
}
This is the preferred way to implement singletons on a JVM because it enables thread-safe lazy initialization without having to rely on a locking algorithm like the complex double-checked locking.
🔗 Source: medium.com
Q7: What will be result of the following code execution?
Topic: Kotlin
Difficulty: ⭐⭐⭐
What will be the output?
val aVar by lazy {
println("I am computing this value")
"Hola"
}
fun main(args: Array<String>) {
println(aVar)
println(aVar)
}
For lazy
the first time you access the Lazy property, the initialisation (lazy()
function invocation) takes place. The second time, this value is remembered and returned:
I am computing this value
Hola
Hola
🔗 Source: dzone.com
Q8: Explain the Null safety in Kotlin
Topic: Kotlin
Difficulty: ⭐⭐⭐
Kotlin's type system is aimed at eliminating the danger of null references from code, also known as the The Billion Dollar Mistake.
One of the most common pitfalls in many programming languages, including Java, is that accessing a member of a null reference will result in a null reference exception. In Java this would be the equivalent of a NullPointerException
or NPE for short.
In Kotlin, the type system distinguishes between references that can hold null
(nullable references) and those that can not (non-null
references). For example, a regular variable of type String can not hold null:
var a: String = "abc"
a = null // compilation error
To allow nulls, we can declare a variable as nullable string, written String?
:
var b: String? = "abc"
b = null // ok
print(b)
🔗 Source: kotlinlang.org
Q9: Explain what is wrong with that code?
Topic: Kotlin
Difficulty: ⭐⭐⭐
Why is this code wrong?
class Student (var name: String) {
init() {
println("Student has got a name as $name")
}
constructor(sectionName: String, var id: Int) this(sectionName) {
}
}
The property of the class can’t be declared inside the secondary constructor.. This will give an error because here we are declaring a property id
of the class in the secondary constructor, which is not allowed.
If you want to use some property inside the secondary constructor, then declare the property inside the class and use it in the secondary constructor:
class Student (var name: String) {
var id: Int = -1
init() {
println("Student has got a name as $name")
}
constructor(secname: String, id: Int) this(secname) {
this.id = id
}
}
🔗 Source: www.programiz.com
Q10: What is a purpose of Companion Objects in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐
Unlike Java or C#, Kotlin doesn’t have static
members or member functions. If you need to write a function that can be called without having a class instance but needs access to the internals of a class, you can write it as a member of a companion object declaration inside that class.
class EventManager {
companion object FirebaseManager {
}
}
val firebaseManager = EventManager.FirebaseManager
The companion object is a singleton. The companion object is a proper object on its own, and can have its own supertypes - and you can assign it to a variable and pass it around. If you're integrating with Java code and need a true static member, you can annotate a member inside a companion object with @JvmStatic
.
🔗 Source: kotlinlang.org
Q11: What is Lateinit in Kotlin and when would you use it?
Topic: Kotlin
Difficulty: ⭐⭐⭐
lateinit means late initialization. If you do not want to initialize a variable in the constructor instead you want to initialize it later on and if you can guarantee the initialization before using it, then declare that variable with lateinit keyword. It will not allocate memory until initialized. You cannot use lateinit for primitive type properties like Int, Long etc.
lateinit var test: String
fun doSomething() {
test = "Some value"
println("Length of string is "+test.length)
test = "change value"
}
There are a handful of use cases where this is extremely helpful, for example:
- Android: variables that get initialized in lifecycle methods;
- Using Dagger for DI: injected class variables are initialized outside and independently from the constructor;
- Setup for unit tests: test environment variables are initialized in a
@Before
- annotated method; - Spring Boot annotations (eg.
@Autowired
).
🔗 Source: medium.com
Q12: When to use lateinit over lazy initialization in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐
There are some simple rules to determined if you should use one or the other for properties initialisation:
- If properties are mutable (i.e. might change at a later stage) use lateInit
- If properties are set externally (e.g. need to pass in some external variable to set it), use lateinit. There’s still workaround to use lazy but not as direct.
- If they are only meant to initialized once and shared by all, and it’s more internally set (dependent on variable internal to the class), then use lazy. Tactically, you could still use lateinit, but using** lazy** would better encapsulate your initialization code.
Also compare:
lateinit var | by lazy |
Can be initialized from anywhere the object seen from. | Can only be initialized from the initializer lambda. |
Multiple initialization possible. | Only initialize single time. |
Non-thread safe. It’s up to user to initialize correctly in a multi-threaded environment. | Thread-safety by default and guarntees that the initializer is invoked by once. |
Can only be used for var. | Can only be used for val. |
Not eligible for nonnull properties. | Not eligible for nonnull properties. |
An isInitialized method added to check whether the value has been initialized before. | Property never able to un-initialized. |
Not allowed on properties of primitive types. | Allowed on properties of primitive types. |
🔗 Source: ahsensaeed.com
Q13: What are coroutines in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐
Unlike many other languages with similar capabilities, async and await are not keywords in Kotlin and are not even part of its standard library.
kotlinx.coroutines
is a rich library for coroutines developed by JetBrains. It contains a number of high-level coroutine-enabled primitives, including launch
, async
and others. Kotlin Coroutines give you an API to write your asynchronous code sequentially.
The documentation says Kotlin Coroutines are like lightweight threads. They are lightweight because creating coroutines doesn’t allocate new threads. Instead, they use predefined thread pools, and smart scheduling. Scheduling is the process of determining which piece of work you will execute next.
Additionally, coroutines can be suspended and resumed mid-execution. This means you can have a long-running task, which you can execute little-by-little. You can pause it any number of times, and resume it when you’re ready again.
🔗 Source: www.raywenderlich.com
Q14: What is the difference between suspending vs. blocking?
Topic: Kotlin
Difficulty: ⭐⭐⭐
A blocking call to a function means that a call to any other function, from the same thread, will halt the parent’s execution. Following up, this means that if you make a blocking call on the main thread’s execution, you effectively freeze the UI. Until that blocking calls finishes, the user will see a static screen, which is not a good thing.
Suspending doesn’t necessarily block your parent function’s execution. If you call a suspending function in some thread, you can easily push that function to a different thread. In case it is a heavy operation, it won’t block the main thread. If the suspending function has to suspend, it will simply pause its execution. This way you free up its thread for other work. Once it’s done suspending, it will get the next free thread from the pool, to finish its work.
🔗 Source: www.raywenderlich.com
Q15: What is the equivalent of Java static methods in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐
Place the function in the companion object.
class Foo {
public static int a() { return 1; }
}
will become:
class Foo {
companion object {
fun a() : Int = 1
}
}
// to run
Foo.a();
Another way is to solve most of the needs for static functions with package-level functions. They are simply declared outside a class in a source code file. The package of a file can be specified at the beginning of a file with the package keyword. Under the hood these "top-level" or "package" functions are actually compiled into their own class. In the above example, the compiler would create a class FooPackage with all of the top-level properties and functions, and route all of your references to them appropriately.
Consider:
package foo
fun bar() = {}
usage:
import foo.bar
🔗 Source: stackoverflow.com
Q16: Explain advantages of "when" vs "switch" in Kotlin
Topic: Kotlin
Difficulty: ⭐⭐⭐
In Java we use switch but in Kotlin, that switch gets converted to when. When has a better design. It is more concise and powerful than a traditional switch. when can be used either as an expression or as a statement.
Some examples of when usage:
- Two or more choices
when(number) {
1 -> println("One")
2, 3 -> println("Two or Three")
4 -> println("Four")
else -> println("Number is not between 1 and 4")
}
- "when" without arguments
when {
number < 1 -> print("Number is less than 1")
number > 1 -> print("Number is greater than 1")
}
- Any type passed in "when"
fun describe(obj: Any): String =
when (obj) {
1 -> "One"
"Hello" -> "Greeting"
is Long -> "Long"
!is String -> "Not a string"
else -> "Unknown"
}
- Smart casting
when (x) {
is Int -> print("X is integer")
is String -> print("X is string")
}
- Ranges
when(number) {
1 -> println("One") //statement 1
2 -> println("Two") //statement 2
3 -> println("Three") //statement 3
in 4..8 -> println("Number between 4 and 8") //statement 4
!in 9..12 -> println("Number not in between 9 and 12") //statement 5
else -> println("Number is not between 1 and 8") //statement 6
}
🔗 Source: blog.mindorks.com
Q17: What are the advantages of Kotlin over Java?
Topic: Kotlin
Difficulty: ⭐⭐⭐
Basically for me less thinking required to write kotlin equivalent to most java code:
data class
- java: you have to write getters and setters for each thing, you have to write
hashCode
properly (or let IDE auto generate, which you have to do again every time you change the class),toString
(same problem ashashcode
) andequals
(same problem ashashCode
). or you could use lombok, but that comes with some quirky problems of its own.record
types are hopefully on the way. *kotlin:data class
does it all for you.
getter and setter patterns
java: rewrite the getter and setter for each variable you use it for
kotlin: don't have to write getter and setter, and custom getter and setter take a lot less typing in kotlin if you do want to. also delegates exist for identical getters\setters
abstract
vs open
classes
java: you have to make an abstract class implementation
kotlin:
open class
lets you make an inheritable class while also being usable itself. nice mix of interface and regular class imo
extension functions
java: doesnt exist
kotlin: does exist, makes functions more clear in usage and feels more natural.
null
java: Anything but primitives can be null at any time.
kotlin: you get to decide what can and cant be null. allows for nice things like
inline class
singleton
java: Memorize singleton pattern
kotlin:
object
instead ofclass
generics
java: Theyre alright, nothing fancy
kotlin: Reified generics (you can access the actual type),
in
andout
for covariance
named parameters
java: does not exist, easy to break api back-compatibility if you arent careful.
kotlin: does exist, easy to preserve api back-compatiblity.
primary constructor
java: does not have per-se, you still have to define everything inside the class
kotlin: very nice to be able to quickly write a constructor without any constructor function or extra needless declarations
🔗 Source: www.reddit.com
Q18: What are some disadvantages of Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐
Some think that Kotlin is a mess of extra syntax and keywords. Here are a few keywords which have non-obvious meanings: internal, crossinline, expect, reified, sealed, inner, open. Java has none of these. Kotlin is also amusingly inconsistent in its keywords: a function is is declared with ‘fun’, but an interface is declared with ‘interface’ (not ‘inter’?). Kotlin also doesn’t have checked exceptions. Checked exceptions have become unfashionable, yet many (including me) find them a powerful way to ensure that your code is robust. Finally, Kotlin hides a lot of what goes on. In Java, you can trace through almost every step of program logic. This can be vital for hunting down bugs. In Kotlin, if you define a data class, then getters, setters, equality testing, to string, and hash code are added for you invisibly. This can be a bad idea.
Also according docs, what Java has that Kotlin does not:
- Checked exceptions
- Primitive types that are not classes
- Static members
- Non-private fields
- Wildcard-types
- Ternary-operator a ? b : c
🔗 Source: www.quora.com
Q19: What is the difference between "open" and "public" in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐
- The open keyword means “open for extension“. The open annotation on a class is the opposite of Java's
final
: it allows others to inherit from this class. - If you do not specify any visibility modifier, public is used by default, which means that your declarations will be visible everywhere. public is the default if nothing else is specified explicitly.
🔗 Source: stackoverflow.com
Q20: What is the difference between “const” and “val”?
Topic: Kotlin
Difficulty: ⭐⭐⭐
const
s are compile time constants. Meaning that their value has to be assigned during compile time, unlike val
s, where it can be done at runtime.
This means, that const
s can never be assigned to a function or any class constructor, but only to a String
or primitive.
For example:
const val foo = complexFunctionCall() //Not okay
val fooVal = complexFunctionCall() //Okay
const val bar = "Hello world" //Also okay
🔗 Source: stackoverflow.com
Q21: What is the difference between List and Array types?
Topic: Kotlin
Difficulty: ⭐⭐⭐
The major difference from usage side is that Arrays
have a fixed size while (Mutable)List
can adjust their size dynamically. Moreover Array
is mutable whereas List
is not.
Furthermore kotlin.collections.List
is an interface implemented among others by java.util.ArrayList
. It's also extended by kotlin.collections.MutableList
to be used when a collections that allows for item modification is needed.
On the jvm level Array
is represented by arrays. List
on the other hand is represented by java.util.List
since there are no immutable collections equivalents available in Java.
🔗 Source: stackoverflow.com
Q22: When would you use Elvis operator in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐
The Elvis operator is part of many programming languages, e.g. Kotlin but also Groovy or C#. The Elvis operator is the ternary operator with its second operand omitted.
x ?: y // yields `x` if `x` is not null, `y` otherwise.
If x
isn't null, then it will be returned. If it is null, then the y
will be returned.
🔗 Source: kotlinlang.org
Q23: Rewrite this code in Kotlin
Topic: Kotlin
Difficulty: ⭐⭐⭐
Can you rewrite this Java code in Kotlin?
public class Singleton {
private static Singleton instance = null;
private Singleton(){
}
private synchronized static void createInstance() {
if (instance == null) {
instance = new Singleton();
}
}
public static Singleton getInstance() {
if (instance == null) createInstance();
return instance;
}
Using Kotlin:
object Singleton
🔗 Source: antonioleiva.com
Q24: How is it recommended to create constants in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐
In Kotlin, if you want to create the local constants which are supposed to be used with in the class then you can create it like below:
val MY_CONSTANT_1 = "Constants1"
// or
const val MY_CONSTANT_2 = "Constants2"
Like val
, variables defined with the const
keyword are immutable. The difference here is that const is used for variables that are known at compile-time.
Also avoid using companion objects. Behind the hood, getter and setter instance methods are created for the fields to be accessible. Calling instance methods is technically more expensive than calling static methods. Instead define the constants in object
:
object DbConstants {
const val TABLE_USER_ATTRIBUTE_EMPID = "_id"
const val TABLE_USER_ATTRIBUTE_DATA = "data"
}
🔗 Source: stackoverflow.com
Q25: May you use IntArray and an Array is in Kotlin interchangeably?
Topic: Kotlin
Difficulty: ⭐⭐⭐
Array<Int>
is an Integer[]
under the hood, while IntArray
is an int[]
.
This means that when you put an Int
in an Array<Int>
, it will always be boxed (specifically, with an Integer.valueOf()
call). In the case of IntArray
, no boxing will occur, because it translates to a Java primitive array.
So no, we can't use them interchangeably.
🔗 Source: stackoverflow.com
Q26: What is the Kotlin double-bang (!!) operator?
Topic: Kotlin
Difficulty: ⭐⭐⭐
The not-null assertion operator !! converts any value to a non-null type and throws a KotlinNullPointerException
exception if the value is null.
Consider:
fun main(args: Array<String>) {
var email: String?
email = null
println(email!!)
}
This operator should be used in cases where the developer is guaranteeing – it allows you to be 100% sure that its value is not null.
🔗 Source: stackoverflow.com
Q27: What is the purpose of Unit-returning in functions? Why is VALUE there? What is this VALUE?
Topic: Kotlin
Difficulty: ⭐⭐⭐
Explain what is the purpose of Unit-returning in functions? Why is VALUE there? What is this VALUE?
fun printHello(name : String?) : Unit {
if (name != null)
print("Hello, $name!")
else
print("Hi there!")
// We don't need to write 'return Unit.VALUE' or 'return', although we could
}
The purpose is the same as C's or Java's void
. Only Unit is a proper type, so it can be passed as a generic argument etc.
Why we don't call it "Void": because the word "void" means "nothing", and there's another type,
Nothing
, that means just "no value at all", i.e. the computation did not complete normally (looped forever or threw an exception). We could not afford the clash of meanings.Why Unit has a value (i.e. is not the same as Nothing): because generic code can work smoothly then. If you pass Unit for a generic parameter T, the code written for any T will expect an object, and there must be an object, the sole value of Unit.
How to access that value of Unit: since it's a singleton object, just say
Unit
UNIT
actually contains valuable information, it basically just means "DONE". It just returns the information to the caller, that the method has been finished.
🔗 Source: stackoverflow.com
Q28: How would you refactor this code using "apply"?
Topic: Kotlin
Difficulty: ⭐⭐⭐
Consider:
class Message(message: String, signature: String) {
val body = MessageBody()
init {
body.text = message + "\n" + signature
}
}
Do you see any refactoring that could be done?
You can write:
class Message(message: String, signature: String) {
val body = MessageBody().apply {
text = message + "\n" + signature
}
}
🔗 Source: cargocult.dev
Q29: What is inline class in Kotlin and when do we need one? Provide an example.
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐
Sometimes it is necessary for business logic to create a wrapper around some type. However, it introduces runtime overhead due to additional heap allocations. Moreover, if the wrapped type is primitive, the performance hit is terrible, because primitive types are usually heavily optimized by the runtime.
Inline classes provide us with a way to wrap a type, thus adding functionality and creating a new type by itself. As opposed to regular (non-inlined) wrappers, they will benefit from improved performance. This happens because the data is inlined into its usages, and object instantiation is skipped in the resulting compiled code.
inline class Name(val s: String) {
val length: Int
get() = s.length
fun greet() {
println("Hello, $s")
}
}
fun main() {
val name = Name("Kotlin")
name.greet() // method `greet` is called as a static method
println(name.length) // property getter is called as a static method
}
Some notes about inline classes:
- A single property initialized in the primary constructor is the basic requirement of an inline class
- Inline classes allow us to define properties and functions just like regular classes
- Init blocks, inner classes, and backing fields are not allowed
- Inline classes can inherit only from interfaces
- Inline classes are also effectively final
🔗 Source: www.baeldung.com
Q30: What is Coroutine Scope and how is that different from Coroutine Context?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐
Coroutines always execute in some context represented by a value of the CoroutineContext type, defined in the Kotlin standard library. The coroutine context is a set of various elements. The main elements are the Job of the coroutine.
CoroutineScope has no data on its own, it just holds a CoroutineContext. Its key role is as the implicit receiver of the block you pass to
launch
,async
etc.
runBlocking {
val scope0 = this
// scope0 is the top-level coroutine scope.
scope0.launch {
val scope1 = this
// scope1 inherits its context from scope0. It replaces the Job field
// with its own job, which is a child of the job in scope0.
// It retains the Dispatcher field so the launched coroutine uses
// the dispatcher created by runBlocking.
scope1.launch {
val scope2 = this
// scope2 inherits from scope1
}
}
}
You might say that CoroutineScope formalizes the way the CoroutineContext is inherited. You can see how the CoroutineScope mediates the inheritance of coroutine contexts. If you cancel the job in scope1
, this will propagate to scope2
and will cancel the launched job as well.
🔗 Source: stackoverflow.com
Q31: How would you override default getter for Kotlin data class?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐
Given the following Kotlin class:
data class Test(val value: Int)
How would I override the Int getter so that it returns 0
if the value negative?
- Have your business logic that creates the data class alter the value to be 0 or greater before calling the constructor with the bad value. This is probably the best approach for most cases.
- Don't use a
data class
. Use a regularclass
.
class Test(value: Int) {
val value: Int = value
get() = if (field < 0) 0 else field
override fun equals(other: Any?): Boolean {
if (this === other) return true
if (other !is Test) return false
return true
}
override fun hashCode(): Int {
return javaClass.hashCode()
}
}
- Create an additional safe property on the object that does what you want instead of having a private value that's effectively overriden.
data class Test(val value: Int) {
val safeValue: Int
get() = if (value < 0) 0 else value
}
🔗 Source: stackoverflow.com
Q32: How to create empty constructor for data class in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐
If you give default values to all the fields - empty constructor is generated automatically by Kotlin.
data class User(var id: Long = -1,
var uniqueIdentifier: String? = null)
and you can simply call:
val user = User()
Another option is to declare a secondary constructor that has no parameters:
data class User(var id: Long,
var uniqueIdentifier: String?){
constructor() : this(-1, null)
}
🔗 Source: stackoverflow.com
Q33: What are Object expressions in Kotlin and when to use them?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐
Sometimes we need to create an object of some class with slight modification, without explicitly declaring a new subclass for it. Java handles this case with anonymous inner classes. Kotlin uses object expression to achieve the same functionality. We can even create an object expression for an interface or abstract class by just implementing their abstract methods.
It is often used as a substitution to a Java anonymous class:
window.addMouseListener(object : MouseAdapter() {
override fun mouseClicked(e: MouseEvent) {
// ...
}
override fun mouseEntered(e: MouseEvent) {
// ...
}
})
🔗 Source: kotlinlang.org
Q34: What's wrong with that code?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐⭐
Let's say I want to override the Int getter so that it returns 0 if the value negative for the data class. What's bad with that approach?
data class Test(private val _value: Int) {
val value: Int
get() = if (_value < 0) 0 else _value
}
The problem with this approach is that data classes aren't really meant for altering data like this. They are really just for holding data. Overriding the getter for a data class like this would mean that Test(0)
and Test(-1)
wouldn't equal one another and would have different hashCodes
, but when you called .value
, they would have the same result. This is inconsistent, and while it may work for you, other people on your team who see this is a data class, may accidentally misuse it.
🔗 Source: stackoverflow.com
Q35: What is the difference between “*” and “Any” in Kotlin generics?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐⭐
-
List<*>
can contain objects of any type, but only that type, so it can containStrings
(but onlyStrings
) - while
List<Any>
can containStrings
andIntegers
and whatnot, all in the same list
🔗 Source: stackoverflow.com
Q36: Imagine you moving your code from Java to Kotlin. How would you rewrite this code in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐⭐
public class Foo {
private static final Logger LOG = LoggerFactory.getLogger(Foo.class);
}
Use Static-like approach:
class MyClass {
companion object {
val LOG = Logger.getLogger(MyClass::class.java.name)
}
fun foo() {
LOG.warning("Hello from MyClass")
}
}
🔗 Source: stackoverflow.com
Q37: How Kotlin coroutines are better than RxKotlin/RxJava?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐⭐
Kotlin coroutines are different from Rx. Both are designed to address a problem of asynchronous programming, however their approach to solution is very different:
Rx comes with a particular functional style of programming that can be implemented in virtually any programming language without support from the language itself. It works well when the problem at hand easily decomposes into a sequence of standard operators and not so well otherwise.
Kotlin coroutines provide a language feature that let library writers implement various asynchronous programming styles, including, but not limited to functional reactive style (Rx). With Kotlin coroutines you can also write your asynchronous code in imperative style, in promise/futures-based style, in actor-style, etc.
How Kotlin coroutines are better than RxKotlin? You just write sequential code, everything is as easy as writing synchronous code except it execute asynchronously. It's easier to grasp.
Coroutines are better to deal with resources
- In RxJava you can assign computations to schedulers but
subscribeOn()
andObserveOn()
are confusing. Every coroutine is given a thread context and return to parent context. For a channel, both side (producer, consumer) execute on his own context. Coroutines are more intuitive on thread or thread pool affectation. - Coroutines give more control on when those computation occur. You can for example pass hand (
yield
), prioritize (select
), parallelize (multipleproducer
/actor
onchannel
) or lock resource (Mutex
) for a given computation. It may not matter on server (where RxJava came first) but on resources limited environment this level of control may be required. - Due to it's reactive nature, backpressure doesn't fit well in RxJava. In the other end
send()
to channel is a suspensive function that suspend when channel capacity is reached. It's out-of-the-box backpressure given by nature. You could alsooffer()
to channel, in which case the call never suspend but returnfalse
in case the channel is full, effectively reproducingonBackpressureDrop()
from RxJava. Or you could just write your own custom backpressure logic, which won't be difficult with coroutines, especially compared to do the same with RxJava.
🔗 Source: stackoverflow.com
Q38: What is the difference between launch/join and async/await in Kotlin coroutines?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐⭐
launch is used to** fire and forget coroutine**. It is like starting a new thread. If the code inside the
launch
terminates with exception, then it is treated like uncaught exception in a thread -- usually printed to stderr in backend JVM applications and crashes Android applications.join
is used to wait for completion of the launched coroutine and it does not propagate its exception. However, a crashed child coroutine cancels its parent with the corresponding exception, too.async is used to start a coroutine that computes some result. The result is represented by an instance of
Deferred
and you must useawait
on it. An uncaught exception inside the async code is stored inside the resultingDeferred
and is not delivered anywhere else, it will get silently dropped unless processed. You MUST NOT forget about the coroutine you’ve started with async.
🔗 Source: stackoverflow.com
Q39: What is The Billion Dollar Mistake?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐⭐
Kotlin's type system is aimed at eliminating the danger of null references from code, also known as the The Billion Dollar Mistake.
One of the most common pitfalls in many programming languages, including Java, is that accessing a member of a null reference will result in a null reference exception. In Kotlin, the type system distinguishes between references that can hold null (nullable references) and those that can not (non-null references).
I call it my billion-dollar mistake. It was the invention of the null reference in 1965. At that time, I was designing the first comprehensive type system for references in an object-oriented language (ALGOL W). My goal was to ensure that all use of references should be absolutely safe, with checking performed automatically by the compiler. But I couldn’t resist the temptation to put in a null reference, simply because it was so easy to implement. This has led to innumerable errors, vulnerabilities, and system crashes, which have probably caused a billion dollars of pain and damage in the last forty years.
Tony Hoare at QCon London in 2009 https://en.wikipedia.org/wiki/Tony_Hoare
🔗 Source: kotlinlang.org
Q40: How to implement Builder pattern in Kotlin?
Topic: Kotlin
Difficulty: ⭐⭐⭐⭐⭐
First and foremost, in most cases you don't need to use builders in Kotlin because we have default and named arguments but if you need one use:
class Car( //add private constructor if necessary
val model: String?,
val year: Int
) {
private constructor(builder: Builder) : this(builder.model, builder.year)
class Builder {
var model: String? = null
private set
var year: Int = 0
private set
fun model(model: String) = apply { this.model = model }
fun year(year: Int) = apply { this.year = year }
fun build() = Car(this)
}
}
Usage:
val car = Car.Builder().model("X").build()
🔗 Source: stackoverflow.com
Thanks 🙌 for reading and good luck on your interview!
Please share this article with your fellow devs if you like it!
Check more FullStack Interview Questions & Answers on 👉 www.fullstack.cafe
Top comments (2)
Nice article! But this is not only for Android Developers. It can be for people who want to learn Kotlin to use it in the backend as an example 😃.
great article! thanks for writing it up.