浅谈java反射和自定义注解的综合应用实例

所属分类: 软件编程 / java 阅读数: 25
收藏 0 赞 0 分享

前言

前几天学习了反射和自定义注解,刚好工作中遇到一个小问题:前台传递到后台的必填字段为空,导致不能插入数据库。就是这样一个小问题,让我考虑到是否可以做一个通用的方法,让前台传递过来的必填字段在后台也校验一遍,如果传递为空,则把响应字段返回提示。因此,我考虑的是用注解的方式,在必填字段上面定义,利用反射得到必填字段的字段名,判断是否为空,并返回响应的信息。

需求模拟

假设客户有:姓名,年龄,地址,手机号码,身份证号等信息,而我们是做金融业务,所以关键是看客户的三要素:姓名,身份证号,手机号码。我们要保证前台传递过来的这三个值不为空。
废话不多说,直接上代码。只看红框里面的即可。

目录结构

客户信息类:Customer

这个是个实体类,我们在:姓名,身份证号码,手机号码上都用了我们的自定义注解。
 

package com.dao.chu.po; 
 
/** 
 * 
* <p>Title: Customer</p> 
* <p>Description:客户信息实体 </p> 
 */ 
public class Customer { 
 
  private int id; 
  @IsRequired 
  private String name; // 姓名 
  @IsRequired 
  private String idnum; // 身份证号码 
  @IsRequired 
  private String phone; // 手机号 
  private String sex; // 性别 
  private int age; // 年龄 
  private String address; // 地址 
 
  @Override 
  public String toString() { 
    return "Customer [id=" + id + ", name=" + name + ", idnum=" + idnum 
        + ", phone=" + phone + ", sex=" + sex + ", age=" + age 
        + ", address=" + address + "]"; 
  } 
 
  public int getId() { 
    return id; 
  } 
 
  public void setId(int id) { 
    this.id = id; 
  } 
 
  public String getName() { 
    return name; 
  } 
 
  public void setName(String name) { 
    this.name = name; 
  } 
 
  public String getIdnum() { 
    return idnum; 
  } 
 
  public void setIdnum(String idnum) { 
    this.idnum = idnum; 
  } 
 
  public String getPhone() { 
    return phone; 
  } 
 
  public void setPhone(String phone) { 
    this.phone = phone; 
  } 
 
  public String getSex() { 
    return sex; 
  } 
 
  public void setSex(String sex) { 
    this.sex = sex; 
  } 
 
  public int getAge() { 
    return age; 
  } 
 
  public void setAge(int age) { 
    this.age = age; 
  } 
 
  public String getAddress() { 
    return address; 
  } 
 
  public void setAddress(String address) { 
    this.address = address; 
  } 
   
} 

自定义注解类:IsRequired
 

package com.dao.chu.po;  
import java.lang.annotation.ElementType; 
import java.lang.annotation.Retention; 
import java.lang.annotation.RetentionPolicy; 
import java.lang.annotation.Target; 
 
/** 
 * 
* <p>Title: IsRequired</p> 
* <p>Description: 字段是否必填 </p> 
 */ 
@Retention(value = RetentionPolicy.RUNTIME)  
@Target(value = {ElementType.FIELD})  
public @interface IsRequired  
{  
  /** 
   *  
   
  * <p>Title: isRequired</p> 
  * <p>Description:true:必填 false:非必填 </p> 
  * @return 
   */ 
  boolean isRequired() default true;  
}  

关键工具类:PoUtils

我们在这个类里面主要用了反射的知识,得到带有自定义注解的字段,并取得这个对象的值进行判断

package com.dao.chu.po;  
  
import java.beans.IntrospectionException; 
import java.beans.PropertyDescriptor; 
import java.lang.reflect.Field; 
import java.lang.reflect.InvocationTargetException; 
 
import javax.jws.WebResult; 
 
import com.sun.xml.internal.ws.util.StringUtils; 
  
/** 
 * 
* <p>Title: PoUtils</p> 
* <p>Description:Po操作工具类 </p> 
 */ 
@SuppressWarnings("unused")  
public class PoUtils  
{  
    
  /** 
  * <p>Title: getProperties</p> 
  * <p>Description: 获取javabean属性通用方法 </p> 
  * @param t 
  * @param beanName 
  * @return 
  * @throws IllegalAccessException 
  * @throws IllegalArgumentException 
  * @throws InvocationTargetException 
  * @throws IntrospectionException 
   */ 
  private static <T> Object getProperties(T t, String beanName)  
    throws IllegalAccessException, IllegalArgumentException, InvocationTargetException, IntrospectionException  
  {  
    Object nameValue = new PropertyDescriptor(beanName, t.getClass()).getReadMethod().invoke(t);  
    return nameValue;  
  }  
    
  /** 
  * <p>Title: IsFieldBlank</p> 
  * <p>Description:判断前台传过来的必填字段是否为空 ,不正确则将相应字段返回 </p> 
  * @param t 
  * @return 
  * @throws IllegalAccessException 
  * @throws IllegalArgumentException 
  * @throws InvocationTargetException 
  * @throws IntrospectionException 
   */ 
  public static <T> RespBody IsFieldBlank(T t)  
    throws IllegalAccessException, IllegalArgumentException, InvocationTargetException, IntrospectionException  
  {  
    RespBody respBody = new RespBody(); 
    StringBuffer sb = new StringBuffer();  
      
    Field[] declaredFields = t.getClass().getDeclaredFields();  
    for (Field field : declaredFields)  
    {  
      field.setAccessible(true);  
      String name = field.getName();  
      boolean fieldHasAnno = field.isAnnotationPresent(IsRequired.class);  
      if (fieldHasAnno)  
      {  
        IsRequired annotation = field.getAnnotation(IsRequired.class);  
        boolean required = annotation.isRequired();  
        if (required)  
        {  
          Object value = getProperties(t, name);  
          if (null == value)  
          {  
            sb.append(name + ",");  
          }  
        }  
      }  
    }  
      
    if (null==sb.toString()||"".equals(sb.toString()))  
    {  
      respBody.isSuccess(); 
    }  
      
    respBody.setSuccess(false); 
    respBody.setMsg(sb.toString().substring(0,sb.toString().lastIndexOf(",")) + " is required");  
    return respBody;  
  }  
}  

RespBody:响应实体类

封装了响应的成功失败以及一些信息

 package com.dao.chu.po; 
 
/** 
 * 
* <p>Title: RespBody</p> 
* <p>Description: 响应实体类</p> 
 */ 
public class RespBody  
{  
  private boolean isSuccess = true;  
    
  private String msg;  
    
  private Object data;  
    
  public boolean isSuccess()  
  {  
    return isSuccess;  
  }  
    
  public void setSuccess(boolean isSuccess)  
  {  
    this.isSuccess = isSuccess;  
  }  
    
  public String getMsg()  
  {  
    return msg;  
  }  
    
  public void setMsg(String msg)  
  {  
    this.msg = msg;  
  }  
    
  public Object getData()  
  {  
    return data;  
  }  
    
  public void setData(Object data)  
  {  
    this.data = data;  
  }  
    
  public RespBody(boolean isSuccess, String msg, Object data)  
  {  
    super();  
    this.isSuccess = isSuccess;  
    this.msg = msg;  
    this.data = data;  
  }  
    
  public RespBody(boolean isSuccess, String msg)  
  {  
    super();  
    this.isSuccess = isSuccess;  
    this.msg = msg;  
  }  
    
  public RespBody()  
  {  
  }  
    
  @Override  
  public String toString()  
  {  
    return "ReturnBody [isSuccess=" + isSuccess + ", msg=" + msg + ", data=" + data + "]";  
  }  
}  

测试类:IsRequiredTest

 package com.dao.chu.po; 
 
/** 
 * 
* <p>Title: IsRequiredTest</p> 
* <p>Description: 必填成员变量测试类</p> 
 */ 
public class IsRequiredTest { 
 
  public static void main(String[] args) { 
     
    Customer customer = new Customer(); 
     
    try { 
      //=========第一次不赋值========== 
      RespBody respBody = PoUtils.IsFieldBlank(customer); 
      //不通过则返回提示信息 
      if (!respBody.isSuccess()) { 
        System.out.println("1."+respBody.getMsg()); 
      } 
       
      //=========第二次给姓名赋值========== 
      customer.setName("张三"); 
      respBody = PoUtils.IsFieldBlank(customer); 
      //不通过则返回提示信息 
      if (!respBody.isSuccess()) { 
        System.out.println("2."+respBody.getMsg()); 
      } 
       
    } catch (Exception e) { 
      e.printStackTrace(); 
    } 
  } 
   
   
   
} 

输出结果

第一次三个值都为空,提示三个都是必填的,第二次因为姓名赋值了,所以提示另外两个是必填的,本次实验宣告结束,本人知识有限,若有更好的方法欢迎指正

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

更多精彩内容其他人还在看

Java的面向对象编程基本概念学习笔记整理

这篇文章主要介绍了Java的面向对象编程基本概念学习笔记整理,包括类与方法以及多态等支持面向对象语言中的重要特点,需要的朋友可以参考下
收藏 0 赞 0 分享

Eclipse下编写java程序突然不会自动生成R.java文件和包的解决办法

这篇文章主要介绍了Eclipse下编写java程序突然不会自动生成R.java文件和包的解决办法 的相关资料,需要的朋友可以参考下
收藏 0 赞 0 分享

基于Java实现杨辉三角 LeetCode Pascal's Triangle

这篇文章主要介绍了基于Java实现杨辉三角 LeetCode Pascal's Triangle的相关资料,需要的朋友可以参考下
收藏 0 赞 0 分享

Java中Spring获取bean方法小结

Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架,如何在程序中获取Spring配置的bean呢?下面通过本文给大家介绍Java中Spring获取bean方法小结,对spring获取bean方法相关知识感兴趣的朋友一起学习吧
收藏 0 赞 0 分享

如何计算Java对象占用了多少空间?

在Java中没有sizeof运算符,所以没办法知道一个对象到底占用了多大的空间,但是在分配对象的时候会有一些基本的规则,我们根据这些规则大致能判断出来对象大小,需要的朋友可以参考下
收藏 0 赞 0 分享

剖析Java中的事件处理与异常处理机制

这篇文章主要介绍了Java中的事件处理与异常处理机制,讲解Java是如何对事件或者异常作出响应以及定义异常的一些方法,需要的朋友可以参考下
收藏 0 赞 0 分享

详解Java的Struts2框架的结构及其数据转移方式

这篇文章主要介绍了详解Java的Struts2框架的结构及其数据转移方式,Struts框架是Java的SSH三大web开发框架之一,需要的朋友可以参考下
收藏 0 赞 0 分享

Java封装好的mail包发送电子邮件的类

本文给大家分享了2个java封装好的mail包发送电子邮件的类,并附上使用方法,小伙伴们可以根据自己的需求自由选择。
收藏 0 赞 0 分享

在Java的Struts中判断是否调用AJAX及用拦截器对其优化

这篇文章主要介绍了在Java的Struts中判断是否调用AJAX及用拦截器对其优化的方法,Struts框架是Java的SSH三大web开发框架之一,需要的朋友可以参考下
收藏 0 赞 0 分享

java多线程Future和Callable类示例分享

JAVA多线程实现方式主要有三种:继承Thread类、实现Runnable接口、使用ExecutorService、Callable、Future实现有返回结果的多线程。其中前两种方式线程执行完后都没有返回值,只有最后一种是带返回值的。今天我们就来研究下Future和Callab
收藏 0 赞 0 分享
查看更多