Cooking Tasty code in Kotlin ββPart 1
β Hello developers π, β In this article, Iβll walk you through some basic recipes to cook your code tasty π in Kotlin. Youβre here in the first part of this series. Iβll present some of my views in front of you and I hope youβll like it. This article is basically for the people who are a beginner in Kotlin, want to start development in Kotlin or people who are coming from other programming languages.
Iβm working with Kotlin for 2 years as of now and in these days Iβm actively working with a community of Kotlin developers. In this time span, I always noticed that the community is using Kotlin but itβs not leveraging this beautiful programming language. In short, I would say β Developers are using Kotlin programming language syntax but theyβre writing code as theyβre writing code in JavaπΆ. Thatβs it!. β Kotlin is a really easy or friendly programming language which is expressive and concise, allowing you to express your ideas with less code. This helps in reducing the amount of boilerplate code in your project and there are so many features of Kotlin which canβt be explained in a single article π. β Thatβs enough introduction I guess and now I think we should start talking about Kotlin. So here are some Kotlin-ish concepts which I would recommend to use in your codebase π and letβs take advantage of this superpower programming language. β β
βοΈ Function
β
- Kotlin allows us to do Object Oriented Programming as well as Functional programming. We can use it in both OO and FP styles or mix elements of the two.
- So itβs not necessary to wrap your logic in a class unnecessarily.
- See below code and notice difference π. β β
- Here you can see we extracted methods out of
object MathUtils
. This is just a small example and this is how we can refactor code. Such functions (declared out of class) in Kotlin are resolved as static members in JVM. β
βοΈ Single Expression Functions
β
- As we already discussed that Kotlin provides us with a way to write expressive code.
- If your function is doing only one thing then you can directly write a function using
=
. - See the difference in both of the below code snippet π. β β
Itβs not necessary to mention return type of a function when we use such expression but IMO it makes code more readable for a person whoβs seeing your code for the first time π β
βοΈ Default Argument Functions
β
- In Java, we generally overload functions if we want to allow configurations with different combinations.
- Itβs not necessary in Kotlin because here Default argument comes for help. β β β
- If you see above snippet, youβll notice function
startSomething()
has parameterconfig
as default argument which will be considered if parameter not provided by caller function. - Itβs a very helpful feature where we can allow a developer to configure things. We can even replace Builder pattern using default arguments in Kotlin. We can achieve it using default arguments + named arguments. β
βοΈ Named Arguments Function
β
- Ideally, functions should not have more than 3β4 parameters.
- But if your function has many parameters then thereβs a possibility that wrong value might be assigned to the wrong parameter (as we are humans π). Here named arguments comes to rescue.
- As we discussed in the previous section, we can use functions over Builder pattern in Kotlin.
- Even we can safely change the order of parameters without any conflicts. β β
- As you can see, using named arguments, our code now looks even more readable. We donβt need to see function definition now. We can directly get to know whatβs happening by just looking at the caller function.
- This really makes it easy to configure things and this is how we can use it instead of Builder pattern.
- Default Arguments + Named Arguments = Sweet Code π π β
βοΈ Scope Functions
β
- Scope functions are part of Kotlin standard library functions.
- When you call such a function on an object with a lambda expression provided, it forms a temporary scope.
- In this scope, you can access the object without its name. Such functions are called scope functions. There are five of them:
let
,run
,with
,apply
, andalso
. - These are very helpful utilities which you can also use to chain consecutive code tasks.
- Also, we can take advantage of using scope functions for handling nullability. For example, see this code. ββ
In this code, we used ?
operator on a person
and used fun let {}
which provides a lambda parameter p
(it remains _it_
if not provided explicitly). Then we can safely use that property.
β
let{}
can be also used to obtain some value after processing. For e.g. here we are getting age from the evaluation performed in the body of a lambda.
β
β
- When we want to perform repetitive operations on any specific field which might modify properties of that instance then
apply {}
is best for such scenarios. See below example π. β
β
The body of lambda of function apply{}
provides this
scope of instance on which weβre calling it and returns the same instance which we can use for chaining later.
β
- Thus, here are examples of other scope functions β β Thereβs a lot more we can do with scope functions. Know more about Scope functions here. β
βοΈ Extension Function
β
- This is one of the best features of Kotlin which allows us to extend the functionality of a class without actually inheriting it.
- These functions are resolved statically i.e. they donβt actually modify classes.
- By using this, we can get rid of traditional utility classes. For example, see code π ββ
As you can see, we directly called date.format(βpatternβ)
.
β
As extension function exists, extension properties also exist. Letβs see them.
β
βοΈ Extension Property
β
- As we discussed the extension function, extension property does the same.
- It does not add the actual field in that class. It just creates a getter function under the hood.
- See the example below β
β
Here we created extension properties on Int
which returns binary, octal and hexadecimal.
β
Observe carefully, we have used
get() =
which is invoked everytime when weβll access the field. β
βοΈ Operator Overloading
β
- Yes, we can overload operators in Kotlin for custom types i.e. classes π
- By using it, we can reduce some boilerplate or can improve the readability of code.
- See the code π β
β If you look, the first snippet looks bit confusing but the second snippet looks good and we get a clear idea of whatβs happening. Know more about operator overloading here. β
βοΈ Infix Function
β
- Infix function improves the beauty of our code.
- It allows us to create our own DSLs.
- It can make our code even more readable in the form of simple language. For example, see this π ββ
Did you saw that line? task assignTo user
. Itβs sweet, isnβt it? π
β
Just mark a function as infix
and you can create your own sweetness in your codebase. You can even cook beautiful code by using Extension Function + Infix Function together β¨οΈ.
β
βοΈ Inline Function
β
- As we saw, Higher-order functions in Kotlin are resolved statically and as theyβre stored as an object which might introduce runtime overhead.
- We can reduce this runtime overhead using
inline
function. - When a function is marked as
inline
it actually generates the code from where itβs called. - For example, see this π β
β
Now as you can see, we have marked processList()
as inline
. Now see generated bytecode here π and youβll see that whatever weβve written in inline function is exactly present in main()
function.
β
β
βοΈ Reified type parameters in Inline Function
β
- How to access the type of parameter in Inline function π€.
- reified keyword comes to rescue here.
- See code first π β
As you can see, now the type of class is accessible inside the function, almost as if it were a normal class. Since the function is inlined, no reflection is needed. Otherwise, without reified we might need to use reflections to deserialize that JSON β
βοΈ Typealias
β
- It allows you to specify an alternative name for existing types.
- If the type name is too long you can introduce a different shorter name and use the new one instead.
- For example, see this π, here we used type alias
Features
to shorten the long generic type β
β
βοΈ Import As
β
- If you have created a class with the same name as of another class.
- If we try to use both classes in a single file it would be a clash β οΈ.
- We can use
as
to import specific class with another name. For example, see this π β
β
Here, we have imported java.util.Date
class as JDate
and we can use it by using that name only in this file.
β
Β
β
Yeah! π Thatβs how we cooked tasty code with Kotlin in this first part of this article and I hope you liked this article. In this article, we saw some basic important concepts of the Kotlin. In the next article, weβll see some advanced one. Stay tuned for the next one π.
β
If you liked this article, share it with everyone! π
β
Β
Sharing is caring! β Β
Thank you! π