本文收录于 kotlin入门潜修专题系列,欢迎学习交流。
创作不易,如有转载,还请备注。
写在前面
君子藏器于身,待时而动。——与君共勉。
反射
java中反射占有举足轻重的地位,很多优秀的框架都离不开反射。那么什么是反射?为什么反射被视为java语言具有动态性的关键?kotlin中的反射又是什么样的?这就是本篇文章要阐述的主题。
反射是指,在运行期间能够动态获取类信息的一种机制,java提供了一套api,可以满足这种需求。
为什么说反射使得java具有了动态性?这就需要先了解下什么静态语言,什么是动态语言了。
我们都知道,编程语言大都具备一些共有的行为操作,比如添加新代码、扩展对象或者定义、修改系统类型等等,而动态语言和静态语言最主要的区分就是在上述行为的执行时机上,动态语言会在运行时执行上述行为,而静态语言上述行为则发生在编译期。
对于java而言,其数据类型等在编译期间就已经确定,也无法在运行的时候使用诸如动态的增加代码、扩展对象等操作,所以认为其是静态语言,而反射的出现改变了这种情况。
反射机制可以让java具备在运行时动态获取类信息的能力,包括一些类型元数据信息、泛型信息等,同时可以动态的修改这些编码信息,诸如此类的,所以说反射机制使java具备了动态语言的特性。
java中的反射
上个小节提到,反射机制让java具备了在运行时获取类信息的能力,因此想了解反射,需要先了解下类是什么。
正常我们所说的类其实是指java中的所有类的统称,而在反射机制中,java为我们专门提供了一个类,那就是首字母大写的Class类(具体类型是java.lang.Class),Class类保存了一个普通java类(可以是任何类)对应的元数据信息。示例如下:
//定义了一个Person类
public class Person {
public Person() {
}
private Person(String name) {
}
private int age;
public String name;
public static void nothing() {
}
public int getAge() {
return age;
}
private void test() {
System.out.println("test");
}
}
//测试类
public class Test {
public static void main(String[] args) {
//我们可以通过这种方式获取Person对应的Class类引用
Class clazz = Person.class;
System.out.println("getName : " + clazz.getName());
for (Constructor constructor : clazz.getConstructors()){
System.out.println("constructors: " + constructor);
}
for (Constructor constructor : clazz.getDeclaredConstructors()){
System.out.println("declared constructors: " + constructor);
}
for (Field field : clazz.getDeclaredFields()) {
System.out.println("declared field: " + field);
}
for (Field field : clazz.getFields()) {
System.out.println("field: " + field);
}
for (Method method : clazz.getDeclaredMethods()){
System.out.println("declare method: " + method);
}
for (Method method : clazz.getMethods()){
System.out.println("method: " + method);
}
}
}
上面代码定义了一个普通的类Person,为了便于观察其对应的Class信息,我们在Person中特意定义了多个有不同访问权限组成的成员变量和成员方法,代码执行完后打印如下:
getName : test.Person
constructors: public test.Person()
declared constructors: public test.Person()
declared constructors: private test.Person(java.lang.String)
declared field: private int test.Person.age
declared field: public java.lang.String test.Person.name
field: public java.lang.String test.Person.name
declare method: private void test.Person.test()
declare method: public int test.Person.getAge()
declare method: public static void test.Person.nothing()
method: public int test.Person.getAge()
method: public static void test.Person.nothing()
method: public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException
method: public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException
method: public final void java.lang.Object.wait() throws java.lang.InterruptedException
method: public boolean java.lang.Object.equals(java.lang.Object)
method: public java.lang.String java.lang.Object.toString()
method: public native int java.lang.Object.hashCode()
method: public final native java.lang.Class java.lang.Object.getClass()
method: public final native void java.lang.Object.notify()
method: public final native void java.lang.Object.notifyAll()
结合打印结果很容易知道上述代码的含义,主要是要注意区分一些相似方法的不同点,现将相似方法的区别整理如下(横向区分):
方法 | 方法 | 区分 |
---|---|---|
getConstructors | getDeclaredConstructors | 二者都是用于获取类的构造方法,但getConstructors只能获取公有的构造方法,而getDeclaredConstructors则可以获取该类定义的全部构造方法 |
getFields | getDeclaredFields | 二者是用于获取类中的字段,getFields只能获取公有字段,getDeclaredFields则可以获取全部字段 |
getMethods | getDeclaredMethods | 二者都是用于获取类中定义的方法,getDeclaredMethods只能获取当前类中定义的方法,不包括超类方法,而getMethods则还能获取父类方法 |
上面简单展示了Class的用法,反射正是基于上述信息来完成运行时的一些动态操作的,下面来看个使用反射动态操作类信息的例子。如下所示:
public static void main(String[] args) {
Person person = new Person();
person.setAge(100);
System.out.println("before: " + person.getAge());
Field[] fields = person.getClass().getDeclaredFields();
for (Field field : fields) {
field.setAccessible(true);
if (field.getName().equals("age")) {
try {
field.set(person, 1);
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
}
System.out.println("after: " + person.getAge());
}
}
我们首先来阐述下我们想要完成的工作:我们定义了一个Person类,在其中定义了一个私有的age字段,其默认值是20。现在我们想动态修改其值,但是由于age是私有字段,所以我们无法通过常规的编码来进行修改。那么,此时我们就可以在运行时利用反射机制来解决该问题。
首先,我们遍历Person类的所有成员,把age字段的权限访问符修改为public,然后通过java为我们提供的field对象的set方法完成属性变更,上述代码执行结果打印如下:
before: 20
after: 1
这就是反射的使用案例之一,我们在运行时通过修改类信息来完成我们想要做的工作。反射的使用案例有很多,这个案例只是冰山一角。java相关的反射暂时阐述至此,主要来看下kotlin中的反射。
kotlin中的反射
上节大概阐述了java中的反射,本节将阐述kotlin中的反射机制,kotlin的反射原理和java基本一致,但语法与java中的有点不太一样。
下面,结合案例来看下kotlin关于类信息的一些操作。
首先我们定义一个Person类,如下所示:
open class Person {
private val age = 20
internal var name = "name"
private fun getAge(): Int {
return age
}
fun personName(): String {
return name
}
fun String.charAtIndexInPerson(index: Int): Char {
return ' '
}
val String.lastCharInPerson: String
get() = "last char"
}
接着,我们再定义一个Person类的子类Student,如下所示:
class Student : Person() {
private val grade = "一年级"
val credit = 90
private fun getGrade(): String {
return grade
}
fun studentCredit(): Int {
return credit
}
fun String.charAtIndexInStudent(index: Int): Char {
return ' '
}
val String.lastCharInStudent: String
get() = "last char"
}
这里定义了一个父类,又定义了一个子类,目的是为了后边可以方便的验证不同方法所表达的不同功能。来看下测试代码:
fun main(args: Array) {
val student = Student()
val clazz = student::class
println(clazz)
clazz.memberProperties.forEach {
println("memberProperties: " + it)
}
clazz.declaredMemberProperties.forEach {
println("declaredMemberProperties: " + it)
}
clazz.memberFunctions.forEach {
println("memberFunctions: " + it)
}
clazz.declaredMemberFunctions.forEach {
println("declaredMemberFunctions: " + it)
}
clazz.memberExtensionProperties.forEach {
println("memberExtensionProperties: " + it)
}
clazz.declaredMemberExtensionProperties.forEach {
println("declaredMemberExtensionProperties: " + it)
}
clazz.memberExtensionFunctions.forEach {
println("memberExtensionFunctions: " + it)
}
clazz.declaredMemberExtensionFunctions.forEach {
println("declaredMemberExtensionFunctions: " + it)
}
}
上面代码演示了kotlin为我们暴露的常见的一些获取类信息的接口,当然还有很多获取类其他信息的接口,这里没有罗列出来,可以自己去验证。上面代码执行完成之后,打印如下:
class Student
memberProperties: val Student.credit: kotlin.Int
memberProperties: val Student.grade: kotlin.String
memberProperties: var Person.name: kotlin.String
declaredMemberProperties: val Student.credit: kotlin.Int
declaredMemberProperties: val Student.grade: kotlin.String
memberFunctions: fun Student.getGrade(): kotlin.String
memberFunctions: fun Student.studentCredit(): kotlin.Int
memberFunctions: fun kotlin.Any.equals(kotlin.Any?): kotlin.Boolean
memberFunctions: fun kotlin.Any.hashCode(): kotlin.Int
memberFunctions: fun Person.personName(): kotlin.String
memberFunctions: fun kotlin.Any.toString(): kotlin.String
declaredMemberFunctions: fun Student.getGrade(): kotlin.String
declaredMemberFunctions: fun Student.studentCredit(): kotlin.Int
memberExtensionProperties: val Student.(kotlin.String.)lastCharInStudent: kotlin.String
memberExtensionProperties: val Person.(kotlin.String.)lastCharInPerson: kotlin.String
declaredMemberExtensionProperties: val Student.(kotlin.String.)lastCharInStudent: kotlin.String
memberExtensionFunctions: fun Student.(kotlin.String.)charAtIndexInStudent(kotlin.Int): kotlin.Char
memberExtensionFunctions: fun Person.(kotlin.String.)charAtIndexInPerson(kotlin.Int): kotlin.Char
declaredMemberExtensionFunctions: fun Student.(kotlin.String.)charAtIndexInStudent(kotlin.Int): kotlin.Char
相信大多数人都不会仔细看上面的代码及打印的结果,这里贴出来只是来证明下面我们结论的正确性,所以你完全可以略过上述测试代码,直接来看下面的对比列表(横向比较):
属性 | 属性 | 区分 |
---|---|---|
memberProperties | declaredMemberProperties | 二者都是用于获取类的属性成员信息,但memberProperties用于获取当前类中的所有属性及其超类中的非私有属性,而declaredMemberProperties则只能用于获取该类定义的全部属性 |
memberFunctions | declaredMemberFunctions | 二者都是用于获取类中定义的成员方法,memberFunctions可用于获取公当前类的所有方法以及其超类的非私有成员方法,declaredMemberFunctions则只能获取当前类中的所有成员方法 |
memberExtensionProperties | declaredMemberExtensionProperties | 二者都是用于获取在当前类中定义的扩展属性,memberExtensionProperties可以获取在当前类以及在其父类中定义的扩展属性,而declaredMemberExtensionProperties则只能获取在当前类中定义的属性 |
memberExtensionFunctions | declaredMemberExtensionFunctions | 二者都是用于获取在当前类中定义的扩展方法,memberExtensionFunctions可以获取在当前类以及在其父类中定义的扩展方法,而declaredMemberExtensionFunctions则只能获取在当前类中定义的方法 |
从表格中可以看出,实际上kotlin暴露的获取类信息的接口和java中基本一致,至于一些相似接口的区别完全可以通过其命名进行区分。
但是有一点要特别注意,那就是在kotlin中我们使用:: class这种方法实际上获取的是kotlin中的类信息,其对应的类定义是kotlin.reflect.KClass,而不是java中的java.lang.Class,但很多时候我们需要通过kotlin代码获取其在java语言中所对应的类信息(比如android开发中常见的activity跳转,就需要获取目标类在java语言中对应的Class类),这个时候我们可以通过以下方法获取:
//获取对应的java语言中的类信息,注意,这里是通过Student的
//实例来获取Student对应的类信息的。
val javaClazz = student::class.java//
上面我们获取KClass类信息的时候,是基于类对象来获取的。我们还可以直接通过类名来获取对应的KClass类信息,如下所示:
val javaClazz = Student::class//我们通过类名来获取对应的类信息
KClass
前面我们演示了获取kotlin中类对应的类信息,这些信息都保存在KClass这个类中,本节将探索下这个类。
首先,通过查找源码我们发现KClass位于kotlin.reflect包中,是个接口,如下所示:
package kotlin.reflect
public interface KClass : KDeclarationContainer, KAnnotatedElement, KClassifier {
public val simpleName: String?
override val members: Collection>
//省略其他一些内容...
}
从该接口的定义可以发现,我们常用的获取类成员的入口members、获取类名称的入口simpleName等都在这个接口中,但是并没有找到上面我们用到的诸如declaredFunctions、memberProperties等属性入口,那么我们为什么能使用这些属性?他们到底定义在了哪里?答案是显然的,那就是这些属性是扩展属性!
确实如此,通过编译器生成的KClass.class这个类文件,我们可以发现,这些属性全部是扩展属性,如下所示:
@kotlin.SinceKotlin public val kotlin.reflect.KClass<*>.declaredFunctions: kotlin.collections.Collection> /* compiled code */
@kotlin.SinceKotlin public val kotlin.reflect.KClass<*>.declaredMemberExtensionFunctions: kotlin.collections.Collection> /* compiled code */
//此处省略其他扩展属性的定义...
找到了这些扩展属性的定义之后,更重要的是关注这些属性的类型,因为这在使用反射的时候往往需要用到。分别摘录如下(结合源代码即编译器为我们生成的代码):
//获取该类的所有成员
override val members: Collection>
//获取该类的所有成员属性
@kotlin.SinceKotlin public val kotlin.reflect.KClass.memberProperties: kotlin.collections.Collection> /* compiled code */
//获取该类的所有成员方法
@kotlin.SinceKotlin public val kotlin.reflect.KClass<*>.memberFunctions: kotlin.collections.Collection> /* compiled code */
上面摘录了三个不同类型的属性,一个是获取类所有成员的members,其集合中的元素类型是KCallable<*>,另一个是获取当前类定义的成员属性的memberProperties,其集合中的元素类型是kotlin.reflect.KProperty1
既然members能够获取所有成员,那么显然可以推断KCallable应该是KProperty1以及KFunction的超类型!通过查找源码我们发现,确实如此,如下所示:
//KFunction类型,继承自KCallable
public interface KFunction : KCallable, Function {}
//从下面可以看出,KProperty1继承自KProperty
public interface KProperty1 : KProperty, (T) -> R {}
//而KProperty继承自KCallable
public interface KProperty : KCallable {}
上面代码验证了我们的推断,实际上KCallable可以用于表示任何成员类型。而KFunction
//获取类静态属性成员对应的属性类型
@kotlin.SinceKotlin public val kotlin.reflect.KClass<*>.staticProperties: kotlin.collections.Collection> /* compiled code */
//获取类内定义的扩展属性成员对应的属性类型
@kotlin.SinceKotlin public val kotlin.reflect.KClass.declaredMemberExtensionProperties: kotlin.collections.Collection> /* compiled code */
但这三者都是继承自KProperty,即KProperty表达的就是属性成员类型。
了解具体返回的类型之后,我们就可以做更多的操作了,比如我们可以针对KProperty的属性做做以下判断:
//定义了一个Person类
class Person {
private val age = 20
var name = "name"
open val address = "address"
}
//测试方法
fun main(args: Array) {
var clazz: KClass = Person::class
clazz.memberProperties.forEach {
//这里我们刻意显示将属性对应的类型标识出来
var property = it as KProperty
//在这里可以进行各种判断
println(property.toString() + " isFinal: " + property.isFinal + " isOpen: " + property.isOpen)
}
}
方法的操作也是同样道理了,这里不再阐述。
最后来看一个kotlin中使用反射的经典案例。
动态代理
代理模式应该都比较清楚,代理模式屏蔽了目标对象的实现细节,可分为静态代理和动态代理。静态代理由于其无法动态的进行扩展,只能在编码层次无限增加,所以有很大的局限性。而动态代理则可以进行动态扩展,可以拥有一份代理逻辑,却能完成不同的业务逻辑。
代理模式是使用反射的很好的案例,下面我们通过kotlin代码来实现该模式。
首先,在没有第三方库支持的情况下,代理模式中的被代理者需要实现一个接口,这里我们先来定义一个被代理者,如下所示:
//首先为被代理者定义一个接口
interface IOperation {
fun operation()
}
//然后定义被代理者,实现了IOperation接口
class RealObject : IOperation {
override fun operation() {
println("real object operation...")
}
}
定义完被代理者之后,我们来看下动态代理中最重要的代理逻辑,如下所示:
//代理逻辑代码
class ProxyFactory {
companion object {
fun getInstance(targetObj: Any): Any? {
return Proxy.newProxyInstance(targetObj.javaClass.classLoader, targetObj.javaClass.interfaces, { proxy, method, args ->
println("proxy start, you can do some work here")
val newArgs = args ?: arrayOfNulls(0)
method.invoke(targetObj, *newArgs)
println("proxy end, you can also do some work here")
})
}
}
}
测试代码如下所示:
fun main(args: Array) {
val operation: IOperation = ProxyFactory.getInstance(RealObject()) as IOperation
operation.operation()
}
执行代码,打印如下所示:
proxy start, you can do some work here
real object operation...
proxy end, you can also do some work here
关于上述代码,需要注意以下几点:
- 我们提供了一个工厂类,用于获取代理实例,这个代理实例代理了我们的目标对象即被代理者。
- newProxyInstance方法需要三个参数,第一个是目标类的类加载器,第二个是目标类实现的接口,第三个是InvocationHandler接口,该接口有个invoke方法。
- 代理逻辑实际上就是invoke方法中的逻辑,在这个方法中,我们首先打印了标识代理执行开始的语句,结果利用kotlin反射机制调用了被代理者的目标方法,最后我们打印了标识代理执行结束的语句。从这个流程可以看出,我们既可以在执行被代理者业务逻辑之前做一些工作,也可以在其之后做一些工作,这也是代理模式的另一部分优势!面向切面编程(AOP)就是利用了动态代理这一思想来实现的。
- 在invoke方法中,我们对参数args进行了处理,主要是kotlin中method.invoke方法接收两个参数,第一个就是我们的被代理对象,第二个则是被代理的方法参数,但是由于这个参数是可变参数,而我们传入的实际上是个数组,这如果放在java中,编译器可以帮我们自动处理,但是kotlin中我们还需要显示进行处理,处理方法就是在数组之前加上星号(*),这样数组就被转换为了可变参数。
- 上述提到了method,那么这个到底是什么?实际上这个就是我们的接口方法,这也是动态代理为什么要求目标者实现接口的原因。打印method的值如下所示:
public abstract void proxy.IOperation.operation()
最后,前面我们说过,动态代理能够在一套代理代码下代理各种对象的执行,那么假如现在我们新增一个代理对象,并且要代理该对象中的有参方法,该怎么实现?
很简单,我们首先定义下我们的代理目标,如下所示:
//定义一个接口
interface IOperation2 {
//定义了一个包含有1个入参的方法
fun operation(operationDesc: String)
}
//目标对象
class RealObject2 : IOperation2 {
override fun operation(operationDesc: String) {
println("operation in Real object: " + operationDesc)
}
}
//测试方法
val operation: IOperation2 = ProxyFactory.getInstance(RealObject2()) as IOperation2
operation.operation(" just do it!")
通过上面的代码我们就完成了重新代理一个新对象的需求,我们发现没有对ProxyFactory进行任何变更!这就是动态代理相对于静态代理的优势,上面代码执行完后,打印如下所示:
proxy start, you can do some work here
operation in Real object: just do it!
proxy end, you can also do some work here
至此,本篇文章阐述完毕。