Skip to content
This repository was archived by the owner on Jan 20, 2023. It is now read-only.

Commit c3b2be5

Browse files
authored
Merge pull request #4 from k163377/feature
Improve processing time.
2 parents c4bae22 + 034da25 commit c3b2be5

File tree

9 files changed

+171
-103
lines changed

9 files changed

+171
-103
lines changed

README.md

+4-4
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ class Dst(
4444
### Set alias on map
4545
#### for getter
4646
```kotlin
47-
class Src(@get:PropertyAlias("aliased") val str: String)
47+
class Src(@KGetterAlias("aliased") val str: String)
4848

4949
class Dst(val aliased: String)
5050
```
@@ -53,7 +53,7 @@ class Dst(val aliased: String)
5353
```kotlin
5454
class Src(val str: String)
5555

56-
class Dst(@param:PropertyAlias("str") private val _src: String) {
56+
class Dst(@param:KPropertyAlias("str") private val _src: String) {
5757
val src = _src.someArrangement
5858
}
5959
```
@@ -64,15 +64,15 @@ val srcMap = mapOf("snake_case" to "SnakeCase")
6464

6565
class Dst(val snakeCase: String)
6666

67-
val dst: Dst = Mapper(DataClass::class.primaryConstructor!!) { it.toSnakeCase }.map(src)
67+
val dst: Dst = Mapper(::DataClass) { it.toSnakeCase }.map(src)
6868
```
6969

7070
### Map param to another class
7171

7272
```kotlin
7373
class CreatorClass @SingleArgCreator constructor(val arg: String) {
7474
companion object {
75-
@SingleArgCreator
75+
@KConverter
7676
fun fromInt(arg: Int): CreatorClass {
7777
return CreatorClass(arg.toString)
7878
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
package com.wrongwrong.mapk.annotations
2+
3+
@Target(AnnotationTarget.PROPERTY)
4+
@Retention(AnnotationRetention.RUNTIME)
5+
annotation class KGetterAlias(val value: String)
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
package com.wrongwrong.mapk.annotations
22

3-
@Target(AnnotationTarget.VALUE_PARAMETER, AnnotationTarget.PROPERTY_GETTER)
3+
@Target(AnnotationTarget.VALUE_PARAMETER)
44
@Retention(AnnotationRetention.RUNTIME)
55
annotation class KPropertyAlias(val value: String)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
package com.wrongwrong.mapk.core
2+
3+
import kotlin.reflect.KFunction
4+
import kotlin.reflect.KParameter
5+
import kotlin.reflect.jvm.isAccessible
6+
7+
class KFunctionForCall<T>(private val function: KFunction<T>, instance: Any? = null) {
8+
val parameters: List<KParameter> = function.parameters
9+
private val originalArray: Array<Any?>
10+
val argumentArray: Array<Any?> get() = originalArray.copyOf()
11+
12+
init {
13+
// この関数には確実にアクセスするためアクセシビリティ書き換え
14+
function.isAccessible = true
15+
originalArray = if (instance != null) {
16+
Array(parameters.size) { if (it == 0) instance else null }
17+
} else {
18+
Array(parameters.size) { null }
19+
}
20+
}
21+
22+
fun call(arguments: Array<Any?>): T {
23+
return function.call(*arguments)
24+
}
25+
}

src/main/kotlin/com/wrongwrong/mapk/core/KMapper.kt

+70-80
Original file line numberDiff line numberDiff line change
@@ -1,132 +1,122 @@
11
package com.wrongwrong.mapk.core
22

33
import com.wrongwrong.mapk.annotations.KConstructor
4+
import com.wrongwrong.mapk.annotations.KGetterAlias
45
import com.wrongwrong.mapk.annotations.KPropertyAlias
56
import com.wrongwrong.mapk.annotations.KPropertyIgnore
7+
import java.lang.reflect.Method
68
import kotlin.reflect.KClass
79
import kotlin.reflect.KFunction
8-
import kotlin.reflect.KProperty1
10+
import kotlin.reflect.KParameter
911
import kotlin.reflect.KVisibility
1012
import kotlin.reflect.full.companionObjectInstance
13+
import kotlin.reflect.full.findAnnotation
1114
import kotlin.reflect.full.functions
1215
import kotlin.reflect.full.isSuperclassOf
1316
import kotlin.reflect.full.memberProperties
1417
import kotlin.reflect.full.primaryConstructor
15-
import kotlin.reflect.jvm.isAccessible
18+
import kotlin.reflect.jvm.javaGetter
19+
20+
class KMapper<T : Any> private constructor(
21+
private val function: KFunctionForCall<T>,
22+
propertyNameConverter: (String) -> String = { it }
23+
) {
24+
constructor(function: KFunction<T>, propertyNameConverter: (String) -> String = { it }) : this(
25+
KFunctionForCall(function), propertyNameConverter
26+
)
1627

17-
class KMapper<T : Any>(private val function: KFunction<T>, propertyNameConverter: (String) -> String = { it }) {
1828
constructor(clazz: KClass<T>, propertyNameConverter: (String) -> String = { it }) : this(
1929
getTarget(clazz), propertyNameConverter
2030
)
2131

22-
private val parameters: Set<ParameterForMap<*>> = function.parameters
23-
.map { ParameterForMap.newInstance(it, propertyNameConverter) }
24-
.toSet()
32+
private val parameterMap: Map<String, ParameterForMap<*>> = function.parameters
33+
.filter { it.kind != KParameter.Kind.INSTANCE }
34+
.associate {
35+
(it.findAnnotation<KPropertyAlias>()?.value ?: propertyNameConverter(it.name!!)) to
36+
ParameterForMap.newInstance(it)
37+
}
2538

2639
init {
27-
if (parameters.isEmpty()) throw IllegalArgumentException("This function is not require arguments.")
28-
29-
// private関数に対してもマッピングできなければ何かと不都合があるため、accessibleは書き換える
30-
function.isAccessible = true
40+
if (parameterMap.isEmpty()) throw IllegalArgumentException("This function is not require arguments.")
3141
}
3242

33-
fun map(srcMap: Map<String, Any?>): T {
34-
return parameters.associate {
35-
// 取得した内容がnullでなければ適切にmapする
36-
it.param to srcMap.getValue(it.name)?.let { value ->
37-
mapObject(it, value)
43+
private fun bindParameters(targetArray: Array<Any?>, src: Any) {
44+
src::class.memberProperties.forEach { property ->
45+
val javaGetter: Method? = property.javaGetter
46+
if (javaGetter != null && property.visibility == KVisibility.PUBLIC && property.annotations.none { annotation -> annotation is KPropertyIgnore }) {
47+
parameterMap[property.findAnnotation<KGetterAlias>()?.value ?: property.name]?.let {
48+
// javaGetterを呼び出す方が高速
49+
javaGetter.isAccessible = true
50+
targetArray[it.index] = javaGetter.invoke(src)?.let { value -> mapObject(it, value) }
51+
}
3852
}
39-
}.let { function.callBy(it) }
53+
}
4054
}
4155

42-
fun map(srcPair: Pair<String, Any?>): T = parameters
43-
.single { it.name == srcPair.first }
44-
.let {
45-
function.callBy(mapOf(it.param to srcPair.second?.let { value -> mapObject(it, value) }))
56+
private fun bindParameters(targetArray: Array<Any?>, src: Map<*, *>) {
57+
src.forEach { (key, value) ->
58+
parameterMap[key]?.let { param ->
59+
// 取得した内容がnullでなければ適切にmapする
60+
targetArray[param.index] = value?.let { mapObject(param, it) }
61+
}
4662
}
63+
}
4764

48-
fun map(src: Any): T {
49-
val srcMap: Map<String, KProperty1.Getter<*, *>> =
50-
src::class.memberProperties.filterTargets().associate { property ->
51-
val getter = property.getAccessibleGetter()
65+
private fun bindParameters(targetArray: Array<Any?>, srcPair: Pair<*, *>) {
66+
parameterMap.getValue(srcPair.first.toString()).let {
67+
targetArray[it.index] = srcPair.second?.let { value -> mapObject(it, value) }
68+
}
69+
}
5270

53-
val key = getter.annotations
54-
.find { it is KPropertyAlias }
55-
?.let { (it as KPropertyAlias).value }
56-
?: property.name
71+
fun map(srcMap: Map<String, Any?>): T {
72+
val array: Array<Any?> = function.argumentArray
73+
bindParameters(array, srcMap)
74+
return function.call(array)
75+
}
5776

58-
key to getter
59-
}
77+
fun map(srcPair: Pair<String, Any?>): T {
78+
val array: Array<Any?> = function.argumentArray
79+
bindParameters(array, srcPair)
80+
return function.call(array)
81+
}
6082

61-
return parameters.associate {
62-
// 取得した内容がnullでなければ適切にmapする
63-
it.param to srcMap.getValue(it.name).call(src)?.let { value ->
64-
mapObject(it, value)
65-
}
66-
}.let { function.callBy(it) }
83+
fun map(src: Any): T {
84+
val array: Array<Any?> = function.argumentArray
85+
bindParameters(array, src)
86+
return function.call(array)
6787
}
6888

6989
fun map(vararg args: Any): T {
70-
val srcMap: Map<String, () -> Any?> = listOf(*args)
71-
.map { arg ->
72-
when (arg) {
73-
is Map<*, *> -> arg.entries.associate { (key, value) ->
74-
(key as String) to { value }
75-
}
76-
is Pair<*, *> -> mapOf(arg.first as String to { arg.second })
77-
else -> {
78-
arg::class.memberProperties.filterTargets().associate { property ->
79-
val getter = property.getAccessibleGetter()
80-
81-
val key = getter.annotations
82-
.find { it is KPropertyAlias }
83-
?.let { (it as KPropertyAlias).value }
84-
?: property.name
85-
86-
key to { getter.call(arg) }
87-
}
88-
}
89-
}
90-
}.reduce { acc, map ->
91-
acc + map
92-
}
90+
val array: Array<Any?> = function.argumentArray
9391

94-
return parameters.associate {
95-
// 取得した内容がnullでなければ適切にmapする
96-
it.param to srcMap.getValue(it.name)()?.let { value ->
97-
mapObject(it, value)
92+
listOf(*args).forEach { arg ->
93+
when (arg) {
94+
is Map<*, *> -> bindParameters(array, arg)
95+
is Pair<*, *> -> bindParameters(array, arg)
96+
else -> bindParameters(array, arg)
9897
}
99-
}.let { function.callBy(it) }
100-
}
101-
}
98+
}
10299

103-
private fun Collection<KProperty1<*, *>>.filterTargets(): Collection<KProperty1<*, *>> {
104-
return filter {
105-
it.visibility == KVisibility.PUBLIC && it.annotations.none { annotation -> annotation is KPropertyIgnore }
100+
return function.call(array)
106101
}
107102
}
108103

109-
private fun KProperty1<*, *>.getAccessibleGetter(): KProperty1.Getter<*, *> {
110-
// アクセス制限の有るクラスではpublicなプロパティでもゲッターにアクセスできない場合が有るため、アクセス可能にして使う
111-
getter.isAccessible = true
112-
return getter
113-
}
114-
115104
@Suppress("UNCHECKED_CAST")
116-
internal fun <T : Any> getTarget(clazz: KClass<T>): KFunction<T> {
117-
val factoryConstructor: List<KFunction<T>> =
105+
internal fun <T : Any> getTarget(clazz: KClass<T>): KFunctionForCall<T> {
106+
val factoryConstructor: List<KFunctionForCall<T>> =
118107
clazz.companionObjectInstance?.let { companionObject ->
119108
companionObject::class.functions
120109
.filter { it.annotations.any { annotation -> annotation is KConstructor } }
121-
.map { KFunctionWithInstance(it, companionObject) as KFunction<T> }
110+
.map { KFunctionForCall(it, companionObject) as KFunctionForCall<T> }
122111
} ?: emptyList()
123112

124-
val constructors: List<KFunction<T>> = factoryConstructor + clazz.constructors
113+
val constructors: List<KFunctionForCall<T>> = factoryConstructor + clazz.constructors
125114
.filter { it.annotations.any { annotation -> annotation is KConstructor } }
115+
.map { KFunctionForCall(it) }
126116

127117
if (constructors.size == 1) return constructors.single()
128118

129-
if (constructors.isEmpty()) return clazz.primaryConstructor!!
119+
if (constructors.isEmpty()) return KFunctionForCall(clazz.primaryConstructor!!)
130120

131121
throw IllegalArgumentException("Find multiple target.")
132122
}

src/main/kotlin/com/wrongwrong/mapk/core/ParameterForMap.kt

+3-13
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
package com.wrongwrong.mapk.core
22

33
import com.wrongwrong.mapk.annotations.KConverter
4-
import com.wrongwrong.mapk.annotations.KPropertyAlias
54
import kotlin.reflect.KClass
65
import kotlin.reflect.KFunction
76
import kotlin.reflect.KParameter
@@ -11,16 +10,7 @@ import kotlin.reflect.full.isSubclassOf
1110
import kotlin.reflect.full.staticFunctions
1211
import kotlin.reflect.jvm.isAccessible
1312

14-
internal class ParameterForMap<T : Any> private constructor(
15-
val param: KParameter,
16-
val clazz: KClass<T>,
17-
propertyNameConverter: (String) -> String
18-
) {
19-
val name: String = param.annotations
20-
.find { it is KPropertyAlias }
21-
?.let { (it as KPropertyAlias).value }
22-
?: propertyNameConverter(param.name!!)
23-
13+
internal class ParameterForMap<T : Any> private constructor(val index: Int, val clazz: KClass<T>) {
2414
val javaClazz: Class<T> by lazy {
2515
clazz.java
2616
}
@@ -34,8 +24,8 @@ internal class ParameterForMap<T : Any> private constructor(
3424
creators.find { (key, _) -> input.isSubclassOf(key) }?.second
3525

3626
companion object {
37-
fun newInstance(param: KParameter, propertyNameConverter: (String) -> String): ParameterForMap<*> {
38-
return ParameterForMap(param, param.type.classifier as KClass<*>, propertyNameConverter)
27+
fun newInstance(param: KParameter): ParameterForMap<*> {
28+
return ParameterForMap(param.index, param.type.classifier as KClass<*>)
3929
}
4030
}
4131
}

src/test/kotlin/mapk/core/GetTargetTest.kt

+15-3
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,12 @@
33
package mapk.core
44

55
import com.wrongwrong.mapk.annotations.KConstructor
6+
import com.wrongwrong.mapk.core.KFunctionForCall
67
import com.wrongwrong.mapk.core.getTarget
8+
import kotlin.reflect.KFunction
9+
import kotlin.reflect.full.memberProperties
710
import kotlin.reflect.full.primaryConstructor
11+
import kotlin.reflect.jvm.isAccessible
812
import org.junit.jupiter.api.Assertions.assertEquals
913
import org.junit.jupiter.api.Assertions.assertTrue
1014
import org.junit.jupiter.api.DisplayName
@@ -27,26 +31,34 @@ class MultipleConstructorDst @KConstructor constructor(val argument: Int) {
2731
@KConstructor constructor(argument: String) : this(argument.toInt())
2832
}
2933

34+
@Suppress("UNCHECKED_CAST")
3035
@DisplayName("クラスからのコンストラクタ抽出関連テスト")
3136
class GetTargetTest {
37+
private fun <T : Any> KFunctionForCall<T>.getTargetFunction(): KFunction<T> {
38+
return this::class.memberProperties.first { it.name == "function" }.getter.let {
39+
it.isAccessible = true
40+
it.call(this) as KFunction<T>
41+
}
42+
}
43+
3244
@Test
3345
@DisplayName("セカンダリコンストラクタからの取得テスト")
3446
fun testGetFromSecondaryConstructor() {
35-
val function = getTarget(SecondaryConstructorDst::class)
47+
val function = getTarget(SecondaryConstructorDst::class).getTargetFunction()
3648
assertTrue(function.annotations.any { it is KConstructor })
3749
}
3850

3951
@Test
4052
@DisplayName("ファクトリーメソッドからの取得テスト")
4153
fun testGetFromFactoryMethod() {
42-
val function = getTarget(CompanionFactoryDst::class)
54+
val function = getTarget(SecondaryConstructorDst::class).getTargetFunction()
4355
assertTrue(function.annotations.any { it is KConstructor })
4456
}
4557

4658
@Test
4759
@DisplayName("無指定でプライマリコンストラクタからの取得テスト")
4860
fun testGetFromPrimaryConstructor() {
49-
val function = getTarget(ConstructorDst::class)
61+
val function = getTarget(ConstructorDst::class).getTargetFunction()
5062
assertEquals(ConstructorDst::class.primaryConstructor, function)
5163
}
5264

0 commit comments

Comments
 (0)