原理
在程序运行过程中,获取实例对象(被new出来的对象)的成员变量信息、方法信息,某些情况也可以修改信息,这个过程其实就是反射。
获取对象信息的原理其实就是:每个实例对象都可以使用.getClass()获取对应的Class对象,信息都存储在Class这个对象里边。
String name = new String();
Class<? extends String> aClass1 = name.getClass();
通过Class去获取该对象的成员变量
@Test
public void test(){
Integer integer = new Integer(1);
Class<? extends Integer> aClass = integer.getClass();
Field[] fields = aClass.getDeclaredFields();
for (Field f : fields){
System.out.println(f.getName());
}
输出
MIN_VALUE
MAX_VALUE
TYPE
digits
DigitTens
DigitOnes
sizeTable
value
SIZE
BYTES
serialVersionUID
使用反射编写范型数组代码
需求
实现一个方法,传入两个参数,一个是数组,一个是扩容后的长度。返回扩容后的数组。
代码
package main.java.proxy;
import java.lang.reflect.Array;
import java.util.Arrays;
public class Reflect {
public static <T> T[] enlargeArray(T[] oldArray, int length) {
Class<?> componentType = oldArray.getClass().getComponentType();//获取数组类型
int oldLength = Array.getLength(oldArray);
T[] newArray = (T[]) Array.newInstance(componentType, length);
System.arraycopy(oldArray, 0, newArray, 0, oldLength);
return newArray;
}
public static void main(String[] args) {
String[] oldArray = {"1", "2", "3"};
String[] newArray = enlargeArray(oldArray, 5);
System.out.println(Arrays.toString(newArray));
}
}
输出
[1, 2, 3, null, null]
ArrayList的扩容机制
源码
ArrayList.java
public boolean add(E e) {
ensureCapacityInternal(size + 1); // Increments modCount!!
elementData[size++] = e;
return true;
}
一直往后找会到这里
Arrays.java
public static <T,U> T[] copyOf(U[] original, int newLength, Class<? extends T[]> newType) {
@SuppressWarnings("unchecked")
T[] copy = ((Object)newType == (Object)Object[].class)
? (T[]) new Object[newLength]
: (T[]) Array.newInstance(newType.getComponentType(), newLength);
System.arraycopy(original, 0, copy, 0,
Math.min(original.length, newLength));
return copy;
反射的应用
IDE的代码提示
比如使用Idea时,键入“.”后,使用反射将可以使用的方法获取到,并显示出来。
读取配置文件
比如mybatis的配置文件里的数据库相关的信息,其实都是通过反射将对象创建,并赋值。
<configuration>
<properties>
<property name="driver" value="com.mysql.cj.jdbc.Driver"/>
<property name="url"
value="jdbc:mysql://127.0.0.1:3306/ssmdemo?useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true"/>
<property name="username" value="root"/>
<property name="password" value="12345678"/>
</properties>
</configuration>