Migration from Kotlin to Java, back to the future

Migration from Kotlin to Java, back to the future

Many projects migrate Java to Kotlin, some go from Kotlin to Java.
These are some notes about my experience in the migration of a Kotlin project to Java. These are technical notes, not an evaluation of the two languages.

🛠️ This post is still in progress. I'm updating it during the migration process.


Cool Features That I'd Like in Java

There are many differences between the two languages. What I miss the most while developing in Java:

  • Nominal parameters
    Many languages have this feature — in Java we don’t and won’t have it. When working with big data objects, it’s easy to write constructor parameters in the wrong order.

  • Copy
    In Kotlin, objects can be copied while modifying some properties:
    👉 Kotlin Data Class Copying


Null Safety

This was a big topic during discussions, but in reality, we didn’t have many problems.

Sure, Java doesn’t come with built-in null safety, but often the issue is exaggerated. Just because a method accepts null doesn’t mean the application logic will pass a null value.

To reassure the team, we allowed the use of Lombok's @NotNull annotation — this helped clarify which parameters could not be null.

For method return types, we preferred to use Optional when nulls were allowed.
With Kotlin interop, using Optional brought some small complexities when integrating with existing functions.


Annoying Small Issues

foreach

  • In Java, return continues to the next item in the loop.
  • In Kotlin, return exits the loop — the equivalent is return@foreach.

typealias

  • In Kotlin, you can define an alias for a type.
  • In Java, this doesn’t exist. We just used the original type.
  • Alternatively, we could create new classes, but that wasn't common in the original project.

Variable – Type Inference

  • We preferred explicit types in Java.
  • When Kotlin code overused type inference, we used var in Java.
  • Our team wanted to see the types rather than guess what kind of object was being used.

Lombok Issues

During the transition, we faced issues with Lombok:

  • Kotlin struggled to interpret Lombok-generated getters as properties.
  • As a result, we had to manually add getters in our Java @Data classes.
  • IntelliJ constantly complained otherwise.

Inline Functions

In Kotlin, small methods were often inlined. For example:

fun findExternalProducts() = 
    productService.findProducts().filter { it.type in ProductType.external }

Migrating these wasn’t too hard — but type inference was the pain point.
When type information is stripped out, the developer has to "think and search" for what kind of data is being manipulated. Easy in simple cases, harder in complex chains.


Kotlin and Jackson

We had to migrate a lot of infrastructure code, including JSON-based components.

Kotlin and Jackson don’t play well together by default:

  • Kotlin data classes don’t have no-arg constructors unless you define default values.
  • Jackson doesn’t know how to map constructor parameters without that.
  • With complex objects, this wasn’t workable.

A workaround:

Add another plugin to make them play nice:

final ObjectMapper objectMapper = new ObjectMapper()
    .registerModule(new KotlinModule.Builder().build());

With Gradle:

implementation("com.fasterxml.jackson.module:jackson-module-kotlin")

Collections

Kotlin has many convenient shorthand methods for collections:

Example: Intersection

collectionOne.intersect(collectionTwo)

Java equivalent:

collectionOne.stream()
    .filter(collectionTwo::contains)
    .collect(Collectors.toSet());

Not too complex, but during migration, you have to stop and think how to "translate" it.

Example: Adding two immutable collections

In Kotlin:

collectionOne + collectionTwo

In Java:

List<Data> allData = new ArrayList<>(collectionOne);
allData.addAll(collectionTwo);
return allData;

Hibernate

There are a lot of online discussions criticizing Hibernate in Kotlin.
We didn’t face many issues. Despite their different philosophies (immutability vs. mutability), they can work well together — especially with the help of 2–3 helper plugins.

Migrating from Kotlin to Java? Hibernate had fewer frictions.


IDE Lock-in

This wasn’t a problem for us, but migrating to Java allowed the frontend team to work more directly with backend code.

Our frontend team used Visual Studio Code (don’t ask me why 😅).
Kotlin has no official support for VS Code, nor plans for it. It’s pretty much a JetBrains-only language.

Unsurprisingly, IntelliJ is very eager to migrate Java → Kotlin.
But the opposite (Kotlin → Java) is not supported.

Some small migration steps

Remove typealias - Easy

In Kotlin we can have typealias like:
typealias Name = String
or
typealias HidingComplexity = Map<Name, Map<Card, Shop>>

After this you can use the type Name to replace String. This, on one side, can help the readability of the code, on the other side, in case of excessive utilisation can create confusion because the types used are not immediately identifiable.

Java doesn't support directly this feature. For the migration to Java we have to remove these aliases and replace them with the original type.

Lombok issue - create getter and setters

We use Lombok in our project, this helps to reduce the ceremonies of the out-of-the-box Java.
With mixed code Kotlin - Java, Kotlin doesn't recognize the Lombok annotations. Probably is possible to have some build configurations that allow a smooth integration of Lombok with Kotlin.

If you have private fields of classes annotated with @Getter or @Data, in Java you can use the standard get method to access the value.

Kotlin automatically remove the get from the method and it tries to access directly the field, doing this it will fail during the compilation because your filed is private and it cannot be accessed.

Kotlin updated to the last version

To simplify the migration it's better to use the latest version of kotlin. Kotlin is moving quickly and maybe some support to feature that you need during the migration can be present in the latest version.
Kotlin will tell you if you are trying to use features that are more recent: e.g. The feature "references to synthetic java properties" is only available since language version 2.1 (we got this when we migrated getter and setters using version 1.9).

enum

This is another easy step, migrate enum class of Kotlin to Java. Pretty straightforward e.g.:

enum class DevSkills(val skill: String) {
  KOTLIN("kotlin");
}

to

import lombok.RequiredArgsConstructor;

@Getter
@RequiredArgsConstructor
public enum DevSkills {
    KOTLIN("kotlin");

    private final String skill;
}
Great! Next, complete checkout for full access to Fullstack developer's journey.
Welcome back! You've successfully signed in.
You've successfully subscribed to Fullstack developer's journey.
Success! Your account is fully activated, you now have access to all content.
Success! Your billing info has been updated.
Your billing was not updated.