一、Java反射的使用场景
临时仅仅用过通过配置文件的className来创建对象。
“非常多人都认为。既然都知道类名,为什么不直接new出来。然后调用方法。搞反射太麻烦了,并且一点用都没有?”对于这个问题,我认为有须要看框架的应该得懂。由于框架实用到,尽管我还没有学习。但在网上看到过这方面问答,所以就冒昧写出来了
二、基础知识
2.1 通过反射得到className的方式 class.forName() 类.class()方法 实例化的类.getClass()方法
2.2 通过ClassName实例化对象的方式 通过class的newInstance的方法 通过Constructor来创建 他们的差别在于。前者仅仅能实例化无參的实例化对象,后者兼容各种參数的实例化对象,当然还包含无參的
2.3 经常使用类 Method跟类的方法挂钩 Field跟类的字段挂钩 Constructor跟类的构造方法挂钩
2.4 经常用法 Method.getMethod()获取public的类方法,局限性非常大 Method.getDeclaredMethod()用法跟上面一样,但除了public方法。还能够获取private方法 假设你尝试使用getMethod()方法来获取private方法,那么将会提示你NoSuchMethodException() 其余的Field方法同上
三、详细实例
实现功能:訪问类的私有方法。并将值打印出来
提供私有方法的类
public class BeiTest1 { public int myself=12; private int getchina(){ return myself; } }
操作类
public class ReflectTest1 { public static void main(String[] args) throws Exception{ BeiTest1 test=new BeiTest1(); //在这里能够使用newInstance或者Constractor来创建实例 Class<?> classType=test.getClass(); //使用getMethod将不能訪问private方法。假设訪问了将会抛出java.lang.NoSuchMethodException异常。兴许也就得不到执行 Method method=classType.getDeclareMethod("getchina",new Class[]{}); //设置是否启用訪问检測,假设为true则表示不检測。也就表示能够訪问 method.setAccessible(true); Integer aaa=(Integer)method.invoke(test, new Object[]{}); System.out.println("结果是"+aaa); } }
实现功能:将某个对象中的字段提取出来。保存到另外一个新建对象中。然后打印出来
class BeiTestCopy{ int age; String name; public BeiTestCopy(int a,String n) { this.age=a; name=n; } }
public class ReflectCopyTest { public static void main(String[] args) throws Exception{ //模拟初始化对象 BeiTestCopy testcopy=new BeiTestCopy(35,"张三"); //调用函数,将通用的Object强转 BeiTestCopy mycopy=(BeiTestCopy) copyObject(testcopy); //打印copy回来的数据 System.out.println(mycopy.name+"今年"+mycopy.age); } public static Object copyObject(Object o) throws Exception{ //获取对象中的各个字段的值 Class<?> classType=o.getClass(); //注,千万要使用declaredFields而非getfields方法。否则无返回 Field[] fields=classType.getDeclaredFields(); //用来存读取到的内容 Object[] obj=new Object[2]; int i=0; for(Field field:fields){ //获取到相应字段的名称 //System.out.println(field.getName()); //field.get接的參数是要提取数据的实例,而非字段名称 //System.out.println("找到的值"+field.get(o)); obj[i]=field.get(o); i++; } //创建对象 Class<?> classType2=o.getClass(); Constructor<?> constructor=classType2.getConstructor(new Class[]{int.class,String.class}); //直接使用先前存放的obj数组就可以,以形成通用模式 //Object newobj=constructor.newInstance(new Object[]{}); Object newobj=constructor.newInstance(obj); //将新创建的对象返回出去 return newobj; } }