反射的概念:JAVA反射机制是在运行状态中,对于任意一个实体类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。

通俗一些 就是说Java通过反射能够创建类的对象,使用类中的方法以及获取类中的属性信息

想写这篇博客缘由是最近做项目时,发现前后台交互,前端(vue)传过来的空对象,后台(springboot)获取后使用 obj == null 是无法判断出来的,其实obj里面的属性对应的属性值都为null, 当对象中嵌套对象再被调用时会抛出空指针异常!!!

eg:

Vue前端发送请求代码:     // 下一步 按钮(转到基本计划页面)

    nextStep: function () {
      var inputForm = {
'proposalStk': {},

SpringBoot 后台部分代码:

    @RequestMapping(value = "/ProposalFlowBasicPlan", method = RequestMethod.POST)
    public  DataResult  ProposalFlowBasicPlan(HttpServletRequest request,
            @RequestBody(required = false) ProposalFlowBasicPlanVO proposalFlowBasicPlanVO) {

       ProposalStk proposalStk = proposalFlowBasicPlanVO.getProposalStk();//从入参中获取 proposalStk 对象,这里获取到的 proposalStk 对象是非 NULL 的

     if(proposalStk != null){

         //if(proposalStk.getStkInsured() != null){  //注释掉这句话时,下面会抛出NullPointerException
            if (onlProposalForm.getOnlProposalFormIsd().getIsdname() != null) {
              proposalStk.getStkInsured().setInsuredName(onlProposalForm.getOnlProposalFormIsd().getIsdname()); 

         //抛空指针异常的原因:proposalStk不为null,但是proposalStk.getStkInsured()为null,再去调用其他方法就会有异常。 StkInsured为对象,不是简单属性。{对象中嵌套对象}
            }

       }

   }    

 

反射分为: 类反射、方法反射、属性反射。

类反射: 通过反射创建类对象。

方法反射:通过反射调用类中的方法信息。

属性反射:通过反射获取类中的属性信息。

 

一、类反射

1、得到类Class对象。

方式一: Class c1 = Student.class; //已知类名时使用此方法
方式二: Class c2 = stu.getClass();//已知该类的对象时使用  
方式三: Class c3 = Class.forName("com.demo.Student");//已知类的全路径名时使用此方法,会有一个ClassNotFoundException异常

2、通过newInstance()方法创建对象的实例。

ps: 注意调用 newInstance()方法创建对象时需要保证 Student类中有个无参数的构造方法!!!

Class c=Class.forName("com.demo.Student");//这个为第一步中的方式三得到Class对象

Student student = (Student)c.newInstance();//创建对象

 

 二、方法反射

1、获取类中的所有方法

   public static void method_1() throws Exception {
      Class clazz = Class.forName("com.demo.Student");
//获取的是该类中所有的公有方法,包含继承和实现的方法,不包含private私有方法。 Method[] methods = clazz.getMethods();
//获取的是该类中的所有方法,包含私有方法,但不包含继承的方法。 methods = clazz.getDeclaredMethods();
for(Method method : methods) { System.out.println(method); } }

 

2、获取类中指定方法名的方法,并且运行该方法

    public static void method_2() throws Exception {
       Class clazz = Class.forName("com.demo.Student");
       //获取指定名称的方法; 参数: 方法名 show, 方法中的入参类型 int.class ,String.class
       Method method = clazz.getMethod("show", int.class,String.class);

Object obj = clazz.newInstance(); method.invoke(obj, 18,"zhagnsan");//执行该方法; 参数: obj为当前类对象, 后面为运行该方法需要的参数值 }

 

3、获取私有方法

        public static void method_3() throws Exception {
             Class clazz = Class.forName("com.makaruina.reflect.Person");
            //想要获取私有方法必须用getDeclearMethod();
             Method method = clazz.getDeclaredMethod("show", null); //当该私有方法时没有参数时使用null.
             // 私有方法不能直接访问,因为权限不够。
             method.setAccessible(true);//一般很少用,因为私有就是隐藏起来,所以尽量不要访问。
        }

 

三、属性反射

public void isObjectFieldEmpty() throws Exception {
         
      Class clazz
=Class.forName("com.demo.Student"); //得到类Class对象 Student student = clazz.newInstance(); Field[] fields = clazz.getDeclaredFields(); //得到所有属性集合 for(Field field : fields ){//遍历属性 field.setAccessible(true); //设置属性是可以访问的(私有的也可以) System.err.println(field.getName()+field.get(student)); // field.getName()得到属性名,field.get(student)得到属性名对应的属性值 } }

 

应用示例:

   /**
     * 想用于对象嵌套对象时的非null判断,避免出现 空指针异常
     * 这里需要注意 序列化 以及带有默认值的字段 布尔类型 boolean 这两种类型是有值的
     * @throws Exception
     */
    @Test
    public void isObjectFieldEmpty() throws Exception {
          CheckboxVO checkVO = new CheckboxVO(); //创建被判断的对象
          checkVO.setCode("123");
          Class clazz=checkVO.getClass();  //得到类Class对象
          System.err.println("CheckVO的Class类对象:"+clazz);
          Field[] fs=clazz.getDeclaredFields(); //得到所有属性集合 
          System.err.println("Fields[]::"+fs);
          List<String> list=new ArrayList<String>();
          for(Field field:fs){            //遍历属性
              field.setAccessible(true); //设置属性是可以访问的(私有的也可以)
              System.err.println("Field::"+field);
              System.err.println("field.get(checkVO):"+field.get(checkVO));
             if(field.get(checkVO)==null||field.get(checkVO)==""){
                String name=(String)field.getName();
                list.add(name);
             }
         }
          System.err.println("list:"+list);
        }

CheckVO.java类

package com.chinalife.proposal.web.build.basicplanVO;

import java.io.Serializable;

public class CheckboxVO implements Serializable{
    
    private static final long serialVersionUID = 1L;

    private String name;

    private boolean check;

    private String code;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public boolean isCheck() {
        return check;
    }

    public void setCheck(boolean check) {
        this.check = check;
    }

    public String getCode() {
        return code;
    }

    public void setCode(String code) {
        this.code = code;
    }
    
}

 

输出结果:

 

 posted on 2018-11-02 16:12  阿叮339  阅读(1109)  评论(0编辑  收藏  举报