Last active
October 24, 2023 09:30
-
-
Save josdejong/fbb43ae33fcdd922040dac4ffc31aeaf to your computer and use it in GitHub Desktop.
Merge two data classes in Kotlin
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import kotlin.reflect.full.declaredMemberProperties | |
import kotlin.reflect.full.primaryConstructor | |
/** | |
* Merge two data classes | |
* | |
* The resulting data class will contain: | |
* - all fields of `other` which are non null | |
* - the fields of `this` for the fields which are null in `other` | |
* | |
* The function is immutable, the original data classes are not changed | |
* and a new data class instance is returned. | |
* | |
* Example usage: | |
* | |
* val a = MyDataClass(...) | |
* val b = MyDataClass(...) | |
* val c = a merge b | |
*/ | |
infix inline fun <reified T : Any> T.merge(other: T): T { | |
val nameToProperty = T::class.declaredMemberProperties.associateBy { it.name } | |
val primaryConstructor = T::class.primaryConstructor!! | |
val args = primaryConstructor.parameters.associateWith { parameter -> | |
val property = nameToProperty[parameter.name]!! | |
(property.get(other) ?: property.get(this)) | |
} | |
return primaryConstructor.callBy(args) | |
} |
Something like this - just change the exact logic to meet your needs (and think through what logic you need for Double, Float, Int. etc, there may be a smart way to check whether any numeric type is zero, I don't know):
infix inline fun <reified T : Any> T.mergeIgnoreZero(other: T): T {
val nameToProperty = T::class.declaredMemberProperties.associateBy { it.name }
val primaryConstructor = T::class.primaryConstructor!!
val args = primaryConstructor.parameters.associateWith { parameter ->
val property = nameToProperty[parameter.name]!!
val valueOther = property.get(other)
val valueThis = property.get(this)
if (valueOther is Double && valueOther == 0.0) {
valueThis
} else {
valueOther ?: valueThis
}
}
return primaryConstructor.callBy(args)
}
@josdejong Thank you very much. It worked.
👍
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
@josdejong Thank you very much. I understand it, but I am new in Kotlin. I am not familiar with
to
operator. I didn't write any inflix function yet. Just I needed this.other
which are non null or non Double with value 0.0this
for the fields which are null or Double with value 0.0 inother
I tried to make it, I am stuck.