一、JavaBean

  * 通过内省引入JavaBean:内省对应的英文全程是IntroSpector。在Java中,其作用主要针对JavaBean进行操作。

  (一)概述

  (1)JavaBean是一种特殊的Java类,主要用于传递数据信息,这种Java类中的方法主要用于访问私有的字段,且方法符合某种特殊的命名规则。

  (2)如果要在两个模块之间传递多个信息,可以将这些信息封装到一个JavaBean中,这种JavaBean的实例对象通常称之为值对象(Value Object,简称VO)

  (3)JavaBean的属性是根据其中的setter和getter方法来确定的,而不是根据其中的成员变量。

  如方法名为setId,则中文意思是设置Id,getId也是如此;去掉set或者get前缀,剩余部分就是属性名称。如果剩余部分的第二个字母小写,则把剩余部分改为小写。如:

  Ø getAge/setAge-->age;

  Ø gettime-->time;

  Ø setTime-->time;

  Ø getCPU-->CPU;

  总之,一个类被当作javaBean使用时,JavaBean的属性是根据方法名推断出来的,它根本看不到java类内部的成员变量。

  (二)JavaBean好处

  符合JavaBean特点的类可以当作普通类一样使用,把它当JavaBean用会带来一些额外的好处:

  1、在Java EE开发中,经常要使用到JavaBean,很多环境就要求按JavaBean方式进行操作。

  2、JDK中提供了对JavaBean进行操作的一些API,这套API就称为内省。用内省这套api操作JavaBean比用普通类的方式更方便。

  (三)内省的简单操作

 

1 import java.beans.BeanInfo;
 2 import java.beans.IntrospectionException;
 3 import java.beans.Introspector;
 4 import java.beans.PropertyDescriptor;
 5 import java.lang.reflect.InvocationTargetException;
 6 import java.lang.reflect.Method;
 7 import java.util.Map;
 8 import org.apache.commons.beanutils.BeanUtils;
 9 import org.apache.commons.beanutils.PropertyUtils;
10 public class IntroSpectorTest {
11     public static void main(String[] args) throws Exception {
12         ReflectPoint pt1 = new ReflectPoint(3,5);
13         String propertyName = "x";        
14         Object retVal = getProperty(pt1, propertyName);    
15         Object value = 7;        
16         setProperties(pt1, propertyName, value);
17         System.out.println(pt1.getX());
18     }
19     private static void setProperties(Object pt1, String propertyName,
20             Object value) throws IntrospectionException,
21             IllegalAccessException, InvocationTargetException {
22         PropertyDescriptor pd2 = new PropertyDescriptor(propertyName,pt1.getClass());
23         Method methodSetX = pd2.getWriteMethod();
24         methodSetX.invoke(pt1,value);
25     }
26     private static Object getProperty(Object pt1, String propertyName)
27             throws IntrospectionException, IllegalAccessException,
28             InvocationTargetException {
29         PropertyDescriptor pd = new PropertyDescriptor(propertyName,pt1.getClass());
30         Method methodGetX = pd.getReadMethod();
31         Object retVal = methodGetX.invoke(pt1);
32         return retVal;
33     }
34 }

  (四)对JavaBean的复杂内省操作

  1、操作步骤:

  (1)在IntroSpector类中有getBeanInfo(Class cls)的方法,通过此方法获取BeanInfo实例。参数是相应对象的字节码,即Class对象。

  (2)BeanInfo类中有getPropertyDescriptors()的方法,可获取所有的JavaBean类中的属性信息,返回一个PropertyDescriptor[]。

  (3)在通过遍历的形式,获取与想要的那个属性信息。

  2、以getX为例,步骤实现的代码示例:

1 private static Object getProperty(Object pt1, String propertyName)
 2             throws IntrospectionException, IllegalAccessException,
 3             InvocationTargetException {        
 4         BeanInfo beanInfo = Introspector.getBeanInfo(pt1.getClass());
 5         PropertyDescriptor[] pds = beanInfo.getPropertyDescriptors();
 6         Object retVal = null;
 7         for(PropertyDescriptor pd : pds){
 8             if(pd.getName().equals(propertyName))
 9             {
10                 Method methodGetX = pd.getReadMethod();
11                 retVal = methodGetX.invoke(pt1);
12                 break;
13             }
14         }
15         return retVal;
16 }

  (五)BeanUtils工具包

  1、beanutils包下载及添加

  (1)下载地址:http://commons.apache.org/proper/commons-beanutils/

  (2)两种添加方式:

  Ø 方式1:项目上“右键”—— Build Path —— Configure Build Path —— 选择Liberiers标签 —— AddExternal Jars —— 选择要导入的jar包。这种方式属于“本地添加”,存在的问题就是如果jar路径发生变化,项目就不能使用到这个jar包。

  Ø 方式2:在项目中建立一个lib目录,专门用于存放项目所使用到的jar工具包。将要使用到jar包复制进来。在Eclipse的Package Explorer面板中,在jar文件上点右键——选择Builder Path——Add to BiuldPath,添加完成。这样做的好处:jar随项目一起移动。

  2、可能出现的问题

  问题:添加beanutils工具包后,可能会出现org/apache/commons/logging/logFactory找不到

  解决方法:下载commons-logging.jar,同beanutils添加步骤一样,添加日志jar包即可。

  3、BeanUtils工具应用,代码示例

1 import org.apache.commons.beanutils.BeanUtils;
 2 import org.apache.commons.beanutils.PropertyUtils;
 3 public class IntroSpectorDemo {
 4     public static void main(String[] args) throws Exception {
 5         reflectPoint rp1 = new reflectPoint(3,5);
 6         String propertyName = "x";
 7         //    用BeanUtils工具包的方法
 8         System.out.println(BeanUtils.getProperty(rp1, "x"));//get
 9         BeanUtils.setProperty(rp1, "x", "9");//set
10         System.out.println(rp1.getX());
11     }
12 }

  4、BeanUtils和PropertyUtils的区别?

  BeanUtils工具包中还有一个工具类PropertyUtils,用法跟BeanUtils一样。区别在于:

  (1)BeanUtils会对JavaBean的属性的类型进行转换,如属性本身是integer,会转换为String。

  (2)PropertyUtils以属性本身的类型进行操作。

1 //用BeanUtils工具包的工具类BeanUtils方法
2 BeanUtils.setProperty(rp1, "x", "9");//set,9是String
3 System.out.println(BeanUtils.getProperty(rp1, "x"));//get
4 System.out.println(BeanUtils.getProperty(rp1, "x").getClass().getName());//java.lang.String
5 
6 //BeanUtils工具包中的PropertyUtils的操作
7 PropertyUtils.setProperty(rp1, "x", 9);//9是Integer
8 System.out.println(PropertyUtils.getProperty(rp1, "x"));
9 System.out.println(PropertyUtils.getProperty(rp1,"x").getClass().getName());//java.lang.Integer

=============================Annotation(注解)==============================

二、注解(Annotation

  (一)概述

  1、注解(Annotation)相当于一种标记,在程序中加了注解就等于为程序打上了某种标记。

  2、标记可以加在包,类,字段,方法,方法的参数以及局部变量上。

  3、格式:@ 注解名称

  (二)Java提供的几个基本注解

  1、@Deprecated:表示告知调用者,该成员函数、字段等已经过时,不再推荐使用。

  (1)让程序部提示函数弃用(过时)?需要在main方法前面,写上@Suppress Warings(“deprecation”)

  (2)注解一个函数过时?在函数的上一行,写上@deprecated

  2、@Override:表示覆盖父类方法。

  (三)注解的应用结构图

Date注解 bean java java中bean注解_apache

  (四)自定义注解及其应用

  1、格式:@interface name{statement},如public @interface MyAnnotation{}

  2、在类上添加注解:

  @MyAnnotation

  public class AnnotationTest{}

  3、元注解(@Retention、@Target)

[雨林木风1]

  (1)RetetionPolicy.SOURSE:java源文件时期,如@Overried和@SuppressWarning

  (2)RetetionPolicy.CLASS: class文件时期(默认阶段)

  (3)RetetionPolicy.RUNTIME:运行时期,如@Deprecated

  如:在某注解类上加@Retention(RetentionPolicy.RUNTIME),表示此注解会一直存在。

  Ø Target:用于说明注解类的使用范围(如:作用在方法上、类上,默认值是任何地方)

  其值可设置为枚举类ElementType类中的任何一个,包括:包、字段、方法、方法参数、构造器、类等值。取值为:

    ANNOTATION_TYPE:注释类型声明;

    CONSIRUCTOR:构造器声明;

    FIELD:字段声明;

    LOCAL_VARIABLE:局部变量声明;

    METHOD:方法声明;

    PACKAGE:包声明;

    PARAMETER:参数声明;

    TYPE:类、接口(包含注释类型)或枚举声明;

  注意:其中代表类的值是TYPE。因为class、enum、interface和@interface等都是平级的,所以统属于Type。不可用CLASS表示。

  4、注解代码示例

1 import java.lang.annotation.ElementType;
 2 import java.lang.annotation.Retention;
 3 import java.lang.annotation.RetentionPolicy;
 4 import java.lang.annotation.Target;
 5 @Retention(RetentionPolicy.RUNTIME) 
 6 @Target({ElementType.METHOD,ElementType.TYPE})
 7 //注解类
 8 @interface MyAnnotation {
 9 }
10 @MyAnnotation
11 public class AnnotationDemo {
12     @SuppressWarnings("deprecation")//此注解用于表明不提示过时信息
13     public static void main(String[] args) {
14         System.runFinalizersOnExit(true); //此方法已过时
15         //判断此类是否有MyAnnotation注解
16         if (AnnotationDemo.class.isAnnotationPresent(MyAnnotation.class)) {
17             //如果有,则获取该注解
18             MyAnnotation annotation = AnnotationDemo.class.getAnnotation(MyAnnotation.class);
19             System.out.println(annotation);
20         }
21     }
22 }

  (五)为注解增加基本属性

  1、注解属性:为注解的参数添加的属性。例如:@MyAnnotation(color="red")

  2、定义基本类型的属性

  (1)在注解类中增加属性:String color();

  (2)定义缺省格式:String value() default “heima;

  3、注解属性应用:

  直接在注解的括号中添加自身的属性:@MyAnnotation(color="red")

  4、value属性:如果注解中有一个名称为value的属性,且只想设置value属性(即其他属性都采用默认值或只有一个value属性),那么可以省略value=部分。例如:

  String value() default "red";

  @MyAnnotation("green")

  5、说明:

  (1)可以为属性值指定缺省值(default),应用时同样可以重新设置属性值。

  (2)用反射方式获得注解对应的实例对象后,可以通过该对象调用属性对应的方法来获取属性值。

  6、代码示例:

1 import java.lang.annotation.Retention;
 2 import java.lang.annotation.RetentionPolicy;
 3 @Retention(RetentionPolicy.RUNTIME)
 4 public @interface MyAnnotation{
 5 String color() default "blue";
 6 String value();
 7 }
 8 
 9 @MyAnnotation(color="red",value="abc")
10 public class AnnotationTest {
11 @SuppressWarnings("deprecation")
12 @MyAnnotation("xyz")
13 public static void main(String[] args) throws Exception{
14     System.runFinalizersOnExit(true);
15     if(AnnotationTest.class.isAnnotationPresentMyAnnotation.class)){
16             System.out.println(annotation.color());
17             System.out.println(annotation.value());
18         }
19     }
20 }

  (六)为注解增加高级属性

  1、数组类型的属性:

  如:int[]arrayArr() default {1,2,3}

  应用:@MyAnnotation(arrayArr={2,3,4})

  注:如果数组属性中只有一个元素,这时候属性值部分可以省略大括。

  2、枚举类型的属性:(定义了一个枚举类TrafficLamp,它是EnumTest的内部类,其值是交通灯的三色)

  定义:EnumTest.TrafficLamplamp();

  应用:@MyAnnotation(lamp=EnumTestTrafficLamp.GREEN)

  3、注解类型的属性:(定义一个注解类:MetaAnnotation,其中定义了一个属性:String value())

  定义:MetaAnnotation annotation() default @MetaAnnotation(”xxx”);

  应用:@MyAnnotation(annotation=@MetaAnnotation(”yyy”))//重新赋值

  4、Class类型的属性:

  定义:Class cls();

  应用:@MyAnnotation(cls=AnnotationDemo.class)

  注:这里的.class必须是已定义的类,或是已有的字节码对象

  5、注解的详细语法可通过查看java语言规范了解即javaLanguage Specification

  6、代码示例:

1 import java.lang.annotation.ElementType;
 2 import java.lang.annotation.Retention;
 3 import java.lang.annotation.RetentionPolicy;
 4 import java.lang.annotation.Target;
 5 @Retention(RetentionPolicy.RUNTIME)//元注释
 6 @Target({ElementType.METHOD,ElementType.TYPE})//元注解,指定使用范围
 7 //注解类
 8 public @interface MyAnnotation {
 9     String color() default "red" ;
10     String value();
11     int[] arr() default {1,2,3};
12     EnumTest.TrafficLamp lamp() default EnumTest.TrafficLamp.GREEN;
13     MetaAnnotation annotation() default @MetaAnnotation("heima");
14     Class clazz() default System.class;
15 }
16 
17 //注解类的应用,给属性赋值或者重新赋值  
18 @MyAnnotation(lamp=EnumTest.TrafficLamp.YELLOW,value="itheima", clazz=AnnotationDemo.class,annotation=@MetaAnnotation("itheima"))
19 public class AnnotationDemo {
20     @SuppressWarnings("deprecation")
21     @MyAnnotation("Method")//自定义注解应用在方法上
22     public static void main(String[] args) throws NoSuchMethodException, SecurityException {
23         System.runFinalizersOnExit(true); //一个过时方法
24         //判断此类是否有MyAnnotation注解
25         if (AnnotationDemo.class.isAnnotationPresent(MyAnnotation.class)) {
26             //如果有,则获取该注解
27             MyAnnotation annotation =AnnotationDemo.class.getAnnotation (MyAnnotation.class);
28             System.out.println(annotation);//@cn.itheima.Demo.MyAnnotation()
29             System.out.println(annotation.color());
30             System.out.println(annotation.value());
31             System.out.println(annotation.arr().length);
32             System.out.println(annotation.lamp());
33             System.out.println(annotation.annotation().value());
34             System.out.println(annotation.clazz());
35         }
36         //获取方法上的注解
37         Method mainMethod = AnnotationDemo.class.getMethod("main",String[].class);
38         MyAnnotation annotationMethod = (MyAnnotation) mainMethod.getAnnotation (MetaAnnotation.class);
39         SuppressWarnings sw=mainMethod.getAnnotation(SuppressWarnings.class);
40         System.out.println(sw);
41         System.out.println(annotationMethod);
42     }
43 }