Java基础(三)-反射
2019-11-05 本文已影响0人
Stan_Z
一、概念
动态获取对象信息和调用对象方法的功能称之为反射。
二、反射的方式
Class.forName() 得到的Class是经过加载、连接和初始化。
Classloder.loadClass() 得到的Class是只进行加载 ,并没有进行连接和初始化。
因此需要初始化的选择Class.forName(),例如数据库连接,在静态块中注册了driver,因此需要初始化。
三、基本使用
public class Person {
public String name = "Stan";
private static int age = 3;
private String gender = "man";
public Person(){
System.out.println("construction method");
}
public String getName(){
return name;
}
private static int getAge(){
return age;
}
private String modifyInfo(String name){
this.name = name;
return name;
}
}
public static void main(String[] args) {
try {
//获取对应的Class
String className = Person.class.getName();
Class clz = Class.forName(className);
//根据双亲委派原则来加载Class.
Class clz = Reflection.class.getClassLoader().loadClass(className);
Class clz = ClassLoader.getSystemClassLoader().loadClass(className);
//直接通过类来获取对应的Class
Class clz = Person.class;
//这几种写法都OK
} catch (Exception e) {
e.printStackTrace();
}
}
//公有属性
public static void getPublicField(Class clz) throws Exception{
Person obj = (Person) clz.newInstance();//创建对象,前提是没有显式构造方法。
//创建对象后可以直接获取其public的属性和方法
System.out.println(obj.name);
System.out.println(obj.getName());
}
//获取带参的公有构造方法
public static void getPublicStructor(Class clz)throws Exception{
Constructor publicConstructor = clz.getDeclaredConstructor(String.class);//先获取构造方法并传参
Person obj = (Person) publicConstructor.newInstance("Milo”);//然后创建对象并赋值,这也是有显示构造方法的初始化方式
System.out.println(obj.name); //同样可以去获取其public的属性和方法
System.out.println(obj.getName());
}
//获取私有属性
//getDeclaredFields() 方法只能获取本类中定义的所有方法(public/protected/default/private),获取不到父类中定义的方法,即使是product的方法也获取不到
//getFields() 方法能获取到本类及父类的方法,但是这些方法必须是public的,其他的都获取不到。
public static void getPrivateField(Class clz)throws Exception{
Object o = clz.newInstance();
//clz.getDeclaredFields() 获取所有属性
Field ageField = clz.getDeclaredField("age”);//获取单个属性
ageField.setAccessible(true);//私有方法需求通过该方法绕过权限
ageField.setInt(o,10);//设置属性
Integer age = (Integer)ageField.get(o);//获取属性
System.out.println(age);
}
//获取带参的私有方法
public static void getPrivateMethod(Class clz) throws Exception{
Object o = clz.newInstance();
//clz.getDeclaredMethods()获取所有方法
Method modifyInfoMethod = clz.getDeclaredMethod("modifyInfo", String.class);//获取当个方法,并传参
modifyInfoMethod.setAccessible(true);
String result = (String) modifyInfoMethod.invoke(o, "Mark”);//方法使用
System.out.println(result);
}
//获取属性类型:field.getGenericType().toString()
//获取属性修饰符(public/private等) field.getModifiers()
非常基础的使用,但是已经很覆盖到大多数的场景了。
四、@hide能否反射获取
这部分还没细致研究,看了一哥们的文章,可信度比较高,后续可以验证下
android9.0后对hide方法反射限制的分析
五、反射的性能问题
Java反射到底慢在哪?
这篇文章有具体的实验测试数据,可信度比较高。从结论来看:耗时方法主要是getMethod、getDeclaredField、invoke和set四个方法,最终调用的都是native方法,猜想有可能是jni调用 外加各种判断逻辑耗时。另外,反射导致的性能问题是否严重跟使用的次数有关系,如果控制在100次以内,基本上没什么差别,如果调用次数超过了100次,性能差异会很明显。