Java基础之反射
2019-08-14

 

如何创建class对象呢?有两种方式:    class.forName("class名称");    Class a = a.class;    两者的区别在于:forName()会帮你初始化好静态变量;    而Class a = a.class这个要自己初始化静态变量;    反射,动态代理;    通过反射,你可以实现获得class的方法;也就是对象的方法;    使用一个newInstace;可以由class变成一个对象;    instanceof可以判断某个类是否是属于另一个类的,用于向下转型时的判断;

 

通过class对象可以获取某个类中的:构造方法,成员变量,成员方法,并访问成员;获取构造方法:    class.getConstructors();获取公有的构造方法;    class.getDeclaredConstructors();获取所有的构造方法;    class.getConstructor(null);获取公有的,无参的构造方法;    这些方法返回的是一个Constructor con;构造函数的类对象;    可以通过con.newInstance("传参")的方式进行 对象的创建;获取成员变量:    class.getFields();获取所有的公有的字段;    class.getDeclaredFields();获取所有的字段;    class.getField("name");获取某一个公有特定字段(name);    class.getDeclaredField("phoneNum");获取某一私有字段phoneNum;    返回Field;用set的方式去设置我们的成员变量;    f.set(obj,"刘德华");其中呢,我们的这个obj就是我们的用class产生的一个对象;    获取成员方法:    class.getMethods();获取所有的公有方法;    class.getDeclaredMethods();获取所有的方法;    class.getMethod("方法名",String.class);获得单个特定的方法;后面的String.class是指我们的方法形参;    返回的是一个Method对象;那我们要做的呢,就是个Field一样,先产生一个对象;    m.invoke(obj,20);这个里面我们用到了方法的执行函数invoke(),obj是对象,20是参数;甚至可以直接反射我们的main()方法;    class.getMethod("main",Stirng[].class);通过反射,我们可以在原有的基础山实现,读取Properties来实现反射,去新建类,以及执行方法;用反射可以避掉泛型的检查:    我们甚至可以调用某一个容器的class文件,再去调用他的method方法,来达到逃避泛型检查的问题; 另外,我们还可以通过反射来实现动态代理的模式:  

interface Subject{ public void doSomething();}class RealSubject implements Subject{ public void doSomething(){ System.out.println("call doSomething()"); }}class ProxyHandler implements InvocationHandler{ private Object tar; public Object bind(Object tar){ this.tar = tar; return Proxy.newProxyInstance(tar.getClass().getClassLoader(), tar.getClass().getInterfaces(), this); } public Object invoke(Object proxy, Method method ,Object[] args) throws Throwable{ Object result = null; result = method.invoke(tar,args); return result; }}public class Test{ public static void main(String[] args[]){ ProxyHandler proxy = new ProxyHandler(); Subject sub = (Subject) proxy.bind(new RealSubject()); sub.doSomething(); }}

其中就用到了一个ProxyHandler,实现自 InvocationHandler接口,通过传入一个真实对象,来的获取一个代理对象,需要的参数有:tar.getClass().getClassLoader()真实对象的类加载器,tar.getClass().getInterfaces()真实对象的接口,以及this,InvocationHandler本身,通过反射的机制,来获取一个代理对象, 同时在这个代理对象的方法执行函数中,写入我们的真实对象的方式执行。以便在之后的方法调用中先调用了我们的真实对象的方法;记得调用代理对象的bind 的时候要把对象向下转成真实对象的;

根据反射获得构造函数,成员变量,成员方法的更多详细代码请参考博客:https://blog.csdn.net/sinat_38259539/article/details/71799078