反射
反射是一种在运行时动态获取和操作类、对象、方法和属性的机制。通过使用反射,你可以在编译时不知道具体类的情况下,获取类的信息、调用方法和访问属性。
在 Java 中,反射机制由 java.lang.reflect 包提供支持。下面是一些常见的反射操作示例:
1.获取类的信息:
Class<?> clazz = MyClass.class; // 通过类字面常量获取
Class<?> clazz = obj.getClass(); // 通过对象实例获取
2.创建对象实例:
Class<?> clazz = MyClass.class;
MyClass instance = (MyClass) clazz.newInstance(); // 创建无参构造函数的实例
3.获取类的构造函数:
Class<?> clazz = MyClass.class;
Constructor<?> constructor = clazz.getConstructor(paramTypes); // 根据参数类型获取特定构造函数
Constructor[] constructors = clazz.getConstructors();//获取所有构造函数
Constructor con1 = clazz.getConstructor(); //获取无参构造函数
Object bean1 = con1.newInstance();
- getConstructors:获取所有构造
- getConstructor(Class… args):获取指定的一个构造函数
- getDeclaredConstructor:获得构造函数,包括private的构造函数
- newInstance(Object… args):执行构造函数产生对象
4.调用对象的方法:
Class<?> clazz = MyClass.class;
Object instance = clazz.newInstance();
Method method = clazz.getMethod("methodName", paramTypes); // 根据方法名和参数类型获取特定方法
method.invoke(instance, args); // 调用方法
- getDeclaredMethods:得到当前类的所有方法,包括private
- getMethods:得到类的所有方法,包括父类的方法
- getMethod(方法名,方法参数列表…):得到指定的方法
- method.invoke(对象实例,方法的参数值列表):调用方法
5.访问对象的属性:
Class<?> clazz = MyClass.class;
Object instance = clazz.newInstance();
Field field = clazz.getDeclaredField("fieldName"); // 获取特定字段
field.setAccessible(true); // 设置可访问私有字段,是违反封装原则
Object value = field.get(instance); // 获取字段值
field.set(instance, newValue); // 设置字段值
- getFields:获得范围内的所有变量(不包括private)
- getDeclaredFields:获得所有变量(包括private)
- getDeclaredField(String name):获得具体的变量(包括private)
- field.getName:获得变量的名字
- field.get(实例):得到变量值
- field.set(实例,变量值):设置变量的值
使用反射时要注意安全性和性能。由于反射是一种动态操作,因此可能会带来一定的性能开销。另外,需要处理异常和访问权限等问题。
如何在不改变String内存地址的情况下,改变String的内容
public void test3() {
String s = new String("abc"); //堆里面产生s
String s1 = s; //s1
s = "123";
String s2 = s;
System.out.println(s1.hashCode() + " " + s2.hashCode() );
System.out.println(s1 == s2);
}
这样改变值,地址就改变了。
为了使地址不改变,我们可以用反射来改变值
public void test4() throws Exception {
String s = new String("123"); //堆里面产生s
String s1 = s; //s1
Field vf = String.class.getDeclaredField("value");
vf.setAccessible(true);
char[] chars = {97, 98, 99};
vf.set(s, chars);
String s2 = s;
System.out.println(s1.hashCode() + " " + s2.hashCode() );
System.out.println(s1 == s2);
}