内省、JavaBean、PropertyDescriptor类、Introspector类、BeanUtils工具包、注解、Rentention、Target、注解的基本属性和高级属性

内省IntroSpector

JavaBean主要用于传递数据信息,其方法用于访问私有变量,且方法名符合某种规则。

如果在两个模块之间传递信息,可以将信息封装进JavaBean中,这种对象称为“值对象”(Value Object),或“VO”。方法比较少。这些信息储存在类的私有变量中,通过set()、get()获得。

内省主要是对JavaBean进行操作。JavaBean内部的方法要按照某种规则命名,例如void setAge(int age)、int getAge()。JavaBean可以作为普通类进行操作;普通类如果内部有set()、get()方法,也可以当做JavaBean使用。

JavaBean的属性是通过get()和set()方法推断出来的,即去掉get、set后的字母,例如,属性为age,而不是成员变量,因为成员变量看不见。

获得属性名的规则:如果属性名的第二个字母是小写,则把第一个字母小写。例如,gettime—>time,setTime—>time,getCPU—>CPU。

JavaBean处理的好处:

1、JavaEE中许多地方需要使用JavaBean。

2、JDK给JavaBean提供的API称为内省。

 

PropertyDescriptor类

PropertyDescriptor类表示JavaBean类通过存储器导出一个属性。主要方法:

1、getPropertyType(),获得属性的Class对象。

2、getReadMethod(),获得用于读取属性值的方法;getWriteMethod(),获得用于写入属性值的方法。

3、hashCode(),获取对象的哈希值。

4、setReadMethod(Method readMethod),设置用于读取属性值的方法;setWriteMethod(MethodwriteMethod),设置用于写入属性值的方法;

导包java.bean.*;

通过属性名获取对应的值,利用反射方法,如下:

ReflectPoint pt1 = new ReflectPoint(7,9);
String propertyName = "x";//给一个属性,获取值
PropertyDescriptor pd = new PropertyDescriptor(propertyName,pt1.getClass());
Method methodGetX = pd.getReadMethod();//Read对应get()方法
Object reValue = methodGetX.invoke(pt1);


给某个属性设置值,如下:

String propertyName2 = "y";//给一个属性,设置值
PropertyDescriptor pd2 = new PropertyDescriptor(propertyName2,pt1.getClass());
Method methodSetY = pd2.getWriteMethod();//Write对应set()方法
methodSetY.invoke(pt1,3);


 

右键—》Source—》Generate Geters and Setters,创建get()和set()方法。

选择一些代码,右键—》Refactor—》Extract Method,创建一个方法,提高复用性。

 

Introspector类

将JavaBean中的属性封装起来进行操作。在程序把一个类当做JavaBean来看,就是调用Introspector.getBeanInfo()方法,得到的BeanInfo对象封装了把这个类当做JavaBean看的结果信息,即属性的信息。需要导包java.beans.*。

getPropertyDescriptors(),获得属性的描述,可以采用遍历BeanInfo的方法,来查找、设置类的属性。

 

private static Object getProperty_2(Object pt1, String propertyName) throws Exception 
{  
    BeanInfo beanInfo = Introspector.getBeanInfo(pt1.getClass());        
    PropertyDescriptor[] pds = beanInfo.getPropertyDescriptors();  
    Object reValue = null; 
  for(PropertyDescriptor pd : pds)  
    {    
      if(pd.getName().equals(propertyName))    
       {    
         Method methodGetX = pd.getReadMethod();
             reValue = methodGetX.invoke(pt1);
             break;
          }
}

return reValue;

}

通过这两个类的比较可以看出,都是需要获得PropertyDescriptor,只是方式不一样:前者通过创建对象直接获得,后者需要遍历,所以使用PropertyDescriptor类更加方便。

 简单的demo

package com.base_super;  
  
import java.beans.BeanInfo;  
import java.beans.IntrospectionException;  
import java.beans.Introspector;  
import java.beans.PropertyDescriptor;  
import java.lang.reflect.InvocationTargetException;  
import java.lang.reflect.Method;  
  
/** 
 * 内省:Introspector 
 *  
 * @author zjw 
 * 
 */  
public class Introspector_class {  
    public static void main(String[] args) throws Exception {  
//      method();  
        methid1();  
    }  
    /* 
     * 简单Introspector实例应用 
     *  
     * 一般情况获取: 
     *      String propertyValue="a"; 
     *      "a"--->"A"--->"getA()"--->再通过反射,method获取值 
     * 通过PropertyDescriptor类,进行反射获取 
     *  
     */  
    public static void method() throws Exception{  
        JavaBean_class bean=new JavaBean_class(3,4);  
        String propertyName="a";//定义JavaBean中的属性  
          
        //get,获取JavaBean中的属性值  
        Object retVal = getProperty(bean, propertyName);//通过选中,单击右键--》Refactor--》Extract Method就可以抽取为方法  
        System.out.println(retVal);  
          
        //set,设置JavaBean中的属性值  
        Object setProperty=88;  
        setPropety(bean, propertyName, setProperty);//同样方法设置,抽取为方法  
        System.out.println(bean.getA());  
          
        //把上面的三句话合并成一句话  
        new PropertyDescriptor(propertyName,bean.getClass()).getWriteMethod().invoke(bean,999);  
        System.out.println(bean.getA());  
    }  
      
    //通过这样抽取为方法,以后可以直接用set/getProperty了,不用老是写,提高代码复用率  
    private static void setPropety(JavaBean_class bean, String propertyName,  
            Object setProperty) throws IntrospectionException,  
            IllegalAccessException, InvocationTargetException {  
        PropertyDescriptor pd1=new PropertyDescriptor(propertyName,bean.getClass());  
        Method m_set=pd1.getWriteMethod();  
        m_set.invoke(bean,setProperty);  
    }  
    //通过这样抽取为方法,以后可以直接用set/getProperty了,不用老是写,提高代码复用率  
    private static Object getProperty(Object bean, String propertyName)  
            throws IntrospectionException, IllegalAccessException,  
            InvocationTargetException {  
        PropertyDescriptor pd=new PropertyDescriptor(propertyName,bean.getClass());//PropertyDescriptor类用来操作JavaBean  
        Method m=pd.getReadMethod();  
        Object retVal=m.invoke(bean);//返回bean这个对象的方法  
        return retVal;  
    }  
      
      
    /* 
     * 复杂Introspector应用实例 
     *  
     * 就是调用Introspector.getBeanInfo方法,得到的BeanInfo对象封装了把这个类当做JavaBean看的结果信息 
     * getBeanInfo方法为Introspector类中的静态方法,所以可以直接调用 
     */  
    public static void methid1() throws Exception{  
        JavaBean_class bean =new JavaBean_class(11,22);  
        String propertyName="a";  
//      BeanInfo bi=Introspector.getBeanInfo(bean.getClass());  
//      PropertyDescriptor[]pds=bi.getPropertyDescriptors();  
//      for (PropertyDescriptor pd : pds) {  
//          if(pd.getName().equals(propertyName)){  
//              Method m=pd.getReadMethod();  
//              m.invoke(bean);  
//              break;    
//          }  
//      }  
        //这个方法就是根据上面注释的语句抽取出来的,为了跟上面的方法不发生冲突,我加了个后缀"_fuza"  
        Object getProperty=getProperty_fuza(bean, propertyName);  
        System.out.println("直接获取属性a的值:"+getProperty);  
          
        Object setProperty=888;  
        setProperty(bean, propertyName, setProperty);  
        System.out.println("设定属性a值后的:"+bean.getA());  
    }  
    //通过抽取出来的,复杂一点的setProperty  
    private static void setProperty(JavaBean_class bean, String propertyName,  
            Object setProperty) throws IntrospectionException,  
            IllegalAccessException, InvocationTargetException {  
        BeanInfo bi_set=Introspector.getBeanInfo(bean.getClass());  
        PropertyDescriptor[] pds_set=bi_set.getPropertyDescriptors();  
        for (PropertyDescriptor pd_set : pds_set) {  
            if(pd_set.getName().equals(propertyName)){  
                Method m_set=pd_set.getWriteMethod();  
                m_set.invoke(bean,setProperty);  
                break;  
            }  
        }  
    }  
    //比上面的方法稍微复杂一点的方式  
    private static Object getProperty_fuza(JavaBean_class bean, String propertyName)  
            throws IntrospectionException, IllegalAccessException,  
            InvocationTargetException {  
        Object getProperty=null;  
        BeanInfo bi=Introspector.getBeanInfo(bean.getClass());  
        PropertyDescriptor[]pds=bi.getPropertyDescriptors();  
        for (PropertyDescriptor pd : pds) {  
            if(pd.getName().equals(propertyName)){  
                Method m=pd.getReadMethod();  
                getProperty=m.invoke(bean);  
                break;  
            }  
        }  
        return getProperty;  
    }  
      
      
}  
class JavaBean_class{  
    private int a;  
    private int b;  
      
    public JavaBean_class(int a,int b){  
        this.a=a;  
        this.b=b;  
    }  
      
    public int getA() {  
        return a;  
    }  
    public void setA(int a) {  
        this.a = a;  
    }  
    public int getB() {  
        return b;  
    }  
    public void setB(int b) {  
        this.b = b;  
    }  
}  


BeanUtils工具包

为JavaBean提供更多、放方便的功能。

beanutils.jar = beanutils-core.jar + beanutils-bean-collections.jar,可以通过BuildPath,添加额外的jar包,或者工程下建立lib目录,将jar包复制进来,再加载这个jar包:右键—》add to BuildPath。使用时需要导包:org.apache.commons.beanutils.BeanUtils。

需要配合使用acpche提供的日志包:logging

获得属性的值,例如,BeanUtils.getProperty(pt1,"x"),返回字符串

设置属性的值,例如,BeanUtils.setProperty(pt1,"y",22),参数是字符串或基本类型自动包装。设置属性的值是字符串,获得的值也是字符串,不是基本类型。

BeanUtils的特点:

1、对基本数据类型的属性的操作:在WEB开发、使用中,录入和显示时,值会被转换成字符串,但底层运算用的是基本类型,这些类型转到动作由BeanUtils自动完成。

2、对引用数据类型的属性的操作:首先在类中必须有对象,不能是null,例如,private Date birthday=new Date();。操作的是对象的属性而不是整个对象,例如,BeanUtils.setProperty(pt1,"birthday.time",121);

Java7的新特性:Map和JavaBean之间可以进行相互转换,key是属性,value是值。

describe:JavaBean—>Map;populate:Map—>JavaBean。例如:

Map map = (name:Kim,age:18);

BeanUtils.setProperty(map,"name","Kim");

  copyProperties(Object dest, Objectorig),将一个对象的属性值复制到另一个对象的属性,需要保证属性一致。

 

PropertyUtils类

和BeanUtils不同在于,运行getProperty、setProperty操作时,没有类型转换,使用属性的原有类型或者包装类。

 

注解Annotation

JDK1.5出现的新特性。在java.lang.annotation包中。

对于过时的语句,java会提示过时了,通过@SuppressWarnings("Deprecation")在DOS中取消提示,但Eclipse无法取消。这就是注解,相当于标记。编译器、开发工具、javac通过反射获得注解里的内容,进而明确应该做什么、不应该做什么。注解可以加在包、类、属性、方法、参数及局部变量之上。

一个注解就是一个类。@SuppressWarnings,取消警告@Deprecated,已过时,老版可以用,新版无法用。

HashSet集合中,对象必须覆盖Object类的equals()方法,否则会继续使用Object类的equals()方法进行比较,错误的比较方法。覆盖equals()方法,参数必须一致,为了防止错误写入本类的对象,加入@Override,必须正确覆盖父类方法,不是创建新方法。

 

注解的应用

在源程序中,调用一个类,这个类会用到注解,需要先准备好注解类,类在调用注解类的对象。注解类的写法类似接口,@interface。先写好注解类A,将注解放在类B中,类C在调用类B时通过反射获得注解类A的内容,进而明确该做什么、不该做什么。可以加上多个注解,加上的实际是注解类的对象:@interfaceA。

main()方法必须放在一个类下,但与这个类不一定有所属关系。

在注解类A上加注解B,这个注解B只为这个注解类A服务,B称为“元注解”。类似的还有元信息、元数据。元注解有2个:Rentention和Target。对注解类的注解,可以理解为注解类的属性。

 

 

Rentention注解类

注解的生命周期:Java源文件—》class文件—》内存中的字节码。编译或者运行时,都有可能会取消注解。Rentention的3种取值意味让注解保留到哪个阶段,RententionPolicy.SOURCE、RententionPolicy.CLASS(默认值)、RententionPolicy.RUNTIME。

@Override、@SuppressWarnings是默认保留到SOURCE阶段;@Deprecated是保留到RUNTIME阶段。

Rentention相当于注解类的一个属性,因为Rentention的值不同,注解类保留到的阶段不同。注解类内部Rentention的值使用value表示,例如,@Deprecated中,value=Runtime。

Rentention的值是枚举RententionPolicy的值,只有3个:SOURCE、CLASS、RUNTIME。

 

Target注解类

性质和Rentention一样,都是注解类的属性,表示注解类应该在什么位置,对那一块的数据有效。例如,@Target(ElementType.METHOD)

Target内部的值使用枚举ElementType表示,表示的主要位置有:注解、构造方法、属性、局部变量、函数、包、参数和类(默认值)。多个位置使用数组,例如,@Target({ElementType.METHOD,ElementType.TYPE})。

类、接口、枚举、注解这一类事物用TYPE表示,Class的父类,JDK1.5的新特性。

 

注解的基本属性

属性,给注解提供更加详细的信息。

注解相当于接口,属性相当于方法。例如,@ItcastAnnotation(color="black"),给属性赋值,取值时类似调用方法,例如,System.out.println(annotation.color());。所有的属性必须全部出现,除非有缺省值。

如果只有value属性,没有其他属性,可以不写=,只针对value,例如,@SuppressWarnings("Deprecation")。或者有其他属性而且有缺省值,例如,String color() default "blue";,此时value单独出现,不用=。

获得注解的属性的值,例如

if(AnnotationDemo.class.isAnnotationPresent(ItcastAnnotation.class)){

ItcastAnnotation annotation =

(ItcastAnnotation)AnnotationDemo.class.getAnnotation(ItcastAnnotation.class);

System.out.println(annotation.color());

}

}

利用反射获得注解的对象,在让该对象调用属性对应的方法。注意类型转换。

Rentention和Target也是属性,都是value对应的值,值的类型分别是RententionPolicy和ElementType,例如,@Retention(value=RetentionPolicy.RUNTIME)。

 

注解的高级属性

给注解增加高级属性,数组、枚举、注解。

数组类型的属性

例如,int[] arr() default {3,7,5};,MyAnnotation(arr={3,7,6})。如果数组只有1个元素,可以不加{}。@Target({ElementType.METHOD,ElementType.TYPE})也是数组类型的属性。

枚举类型的属性

//注解类内部的内容

EnumerationDemo.TrafficLamp lamp() default EnumerationDemo.TrafficLamp.RED;

//调用注解的类上

@ItcastAnnotation(lamp=EnumerationDemo.TrafficLamp.YELLOW)

//对注解进行操作

System.out.println(annotation.lamp().nextLamp().name());

注解类型的属性

将一个注解类作为属性加入到另一个注解类中。

MetaAnnotation annotationAtt() default @MetaAnnotation("Jobs")

@ItcastAnnotation(annotationAtt=@MetaAnnotation("Kim"))

annotation.annotationAtt().value()

注解的返回值可以是8个基本类型、String、Class、枚举以及前面类型的数组,内部还有属性。

需要详细学习注解,可以通过java语言规范,即languagespecification。

你可能感兴趣的:(java)