Kotlin中有以下四种元注解(用来定义注解的注解):
在Kotlin中定义一个注解类,需要使用 annotation 关键字:
@Target(AnnotationTarget.PROPERTY)
annotation class Valid
说明:此次实例是对Spring框架中的Value注解进行简单的实现。大致的运行过程可以概括成这样:在某个类中为它的属性添加@Value(value="key")注解,在配置文件中为注解中出现的关键字赋予相应的值。最终通过注解解析器将配置文件中的值注入到添加了注解的属性中。
说明:我们所定义的是一个属性层级的注解,并且需要在运行时获取注解的相关信息,注解含有一个String类型的参数。最终注解定义的代码是这样子的:
@Target(AnnotationTarget.PROPERTY)
@Retention(AnnotationRetention.RUNTIME) //这一行也可以省略
annotation class Value(val value:String)
说明:注解的定义十分的简单,就那么几行代码。但你要想让注解真正的起作用,你还需要对注解进行相应的解析才行。在解析注解的过程中会使用到大量和反射有关的代码,对反射的概念不熟悉的同学,先去看看反射要不然看接下来的这些代码会蛮吃力的。
class AnnotationExpression (val obj:Any){
fun expression(){
val clazz=obj::class
clazz.declaredMemberProperties.forEach { prop->
val mutableProp= try{
prop as KMutableProperty<*>
}catch (e:Exception){
null
} ?: return@forEach
mutableProp.annotations.forEach { annotation->
val propClassName=mutableProp.returnType.toString().removePrefix("kotlin.")
when(propClassName) {
in numtypeSet->mutableProp.setter.call(obj,
(readProp(annotation as Value) as kotlin.String).toNum(propClassName))
"String"->mutableProp.setter.call(obj,
(readProp(annotation as Value) as kotlin.String))
"Boolean"->mutableProp.setter.call(obj,
(readProp(annotation as Value) as kotlin.String).toBoolean())
}
}
}
}
通过KClass获取的KProperty1默认是不能被修改的,意味着你只能获取属性的值,而不能对其进行修改。所以在这里,我们对它进行了以下转换
prop as KMutableProperty<*>
因为可能出现使用该注解注释val变量的情况,在这里还进行了异常捕获,当发生异常时,直接跳过接下来的处理过程。
我们可以看到上面的代码中多次出现了 readProp 函数,在这里这个函数的作用是根据注解的信息,从配置文件中读取相应的数据。
private fun readProp(value:Value): Any? {
val prop=Properties()
prop.load( AnnotationExpression::class.java.getResource("app.properties").openStream())
return prop.get(value.value)
}
# app.properties
name="feint"
age=11
money=13.5
gender=true
由于直接从property中获取的类型可能会和使用了@Value注解的属性的类型不匹配,因此我们需要根据属性的类型对从配置中获取的类型进行转换。
在这里适配了,布尔型、字符串型以及数字型的数据。由于数字型的类别特别多(Int,Double,Byte等等),便专门为String扩展了一个 toNum 函数,它接受一个String类型的参数,表示类型的名称。具体的代码是下面这样子的:
fun String.toNum(className:String):Any{
val clazz=Class.forName("java.lang.${typeMap[className]}")
return clazz.getMethod("parse$className",String::class.java).invoke(null,this)
}
这个地方又有一个坑,我本来是想通过反射调用Kotlin的String类中类似toInt、toDouble的方法。可是,运行后竟然提示,Kotlin的内置类型目前对反射的支持还不完善。。。无奈只好使用Java中那些包装类的parse方法。
说明:使用的过程也没啥好说的,直接上代码
class User{
@Value(value = "name")
lateinit var name:String
@Value(value = "age")
var age:Int=0
@Value(value = "money")
var money:Double=0.0
@Value(value = "gender")
var gender:Boolean=false
override fun toString(): String {
return "(name:$name; age:$age; money:$money; gender:${if(gender) "man" else "woman"})"
}
}
fun main(args: Array<String>) {
val user=User()
AnnotationExpression(user).expression()
println(user.toString())
}
源码地址(Github):注解练习
欢迎关注Kotlin学习网