JAVA反射

动态语言

动态语言,是指程序在运行时可以改变其结构:新的函数可以引进,已有的函数可以被删除等结构上的变化。

比如常见的JavaScript就是动态语言,除此之位Ruby,Python等也属于动态语言,而C、C++则不属于动态语言。

从反射角度说JAVA属于半动态语言。

 

反射机制概念(运行状态中知道类所有的属性和方法)

在Java中的反射机制是指在运行状态中,对于任意一个类都能知道这个类所有的属性和方法;并且对于任意一个

对象,都能够调用它的任意一个方法;这种动态获取信息以及动态调用对象方法的功能成为Java语言的反射机制。

 

反射应用的场合

编译时类型和运行时类型

在Java程序中许多对象在运行时都会出现两种类型:编译时类型和运行时类型。编译时的类型由声明对象时实用

的类型来决定,运行时的类型由实际赋值给对象的类型决定。如:

Person p = new Student();其中编译时类型为Person,运行时类型为Student。

编译时类型无法获取具体方法

程序在运行时还可能接收到外部传入的对象,该对象的编译时类型为Object,但是程序有需要调用该对象的运行时

类型的方法,程序需要在运行时发现对象和类的真实信息。然而,如果编译时根本无法预知该对象和类属于哪些类,

程序只能依靠运行时信息来发现该对象和类的真实信息,此时就必须使用到反射了。

 

Java反射API

反射API用来生成JVM中的类、接口或者对象的信息。

1.Class类:反射的核心类,可以获取类的属性,方法等信息。

2.Field类:Java.lang.reflec包中的类,表示类的成员变量,可以用来获取和设置类之中的属性值。

3.Method类:Java.lang.reflec包中的类,表示类的方法,它可以用来获取类中的方法信息或者执行方法。

4.Constructor类:Java.lang.reflec包中的类,表示类的构造方法。

 

反射使用步骤(获取Class对象、调用对象方法)

1.获取想要操作的类的Class对象,他是反射的核心,通过Class对象我们可以任意调用类的方法。

2.调用Class类中的方法,既就是反射的使用阶段。

3.使用反射API来操作这些信息。

 

获取Class对象的3种方法

1.调用某个对象的getClass()方法

Person p = new Person();

Class clazz = p.getClass();

2.调用某个类的class属性来获取该类对应的Class对象

Class clazz = Person.class;

3.使用Class类中的forName()静态方法(最安全/性能最好)

Class clazz = Class.forName("类的全路径");(最常用)

当我们获得了想要操作的类的Class对象后,可以通过Class类中的方法获取并查看该类中的方法和属性。

 1 //获取 Person 类的 Class 对象
 2 Class clazz=Class.forName("reflection.Person");
 3 //获取 Person 类的所有方法信息
 4 Method[] method=clazz.getDeclaredMethods();
 5 for(Method m:method){
 6 System.out.println(m.toString());
 7 }
 8 //获取 Person 类的所有成员属性信息
 9 Field[] field=clazz.getDeclaredFields(); for(Field
10 f:field){
11 System.out.println(f.toString());
12 }
13 //获取 Person 类的所有构造方法信息
14 Constructor[] constructor=clazz.getDeclaredConstructors();
15 for(Constructor c:constructor){
16 System.out.println(c.toString());
17 }

 

创建对象的两种方法

Class对象的newInstance()

1.使用Class对象的newInstance()方法来创建该Class对象对应类的实例,但是这种方法要求该Class对象对应的类

有默认的空构造器。

调用Constructor对象的newInstance()

2.先使用Class对象获取指定的Constructor对象,再调用Constructor对象的newInstance()方法来创建Class对象

对应类的实例,通过这种方法可以选定构造方法创建实例。

1 //获取 Person 类的 Class 对象
2 Class clazz=Class.forName("reflection.Person");
3 //使用.newInstane 方法创建对象
4 Person p=(Person) clazz.newInstance();
5 //获取构造方法并创建对象
6 Constructor
7 c=clazz.getDeclaredConstructor(String.class,String.class,int.class);
8 //创建对象并设置属性
9 Person p1=(Person) c.newInstance("李四","男",20);

 

Demo:

1.定义一个自定义注解

 1 /**
 2  * 定义注解
 3  */
 4 @Target(ElementType.FIELD)
 5 @Retention(RetentionPolicy.RUNTIME)
 6 @Documented
 7 public @interface FruitProvider {
 8 
 9     /**
10      * 供应商编号
11      * @return
12      */
13     public int id() default -1;
14 
15     /**
16      * 供应商名称
17      * @return
18      */
19     public String name() default "";
20 
21     /**
22      * 供应商地址
23      * @return
24      */
25     public String address() default "";
26 
27 }

2.定义一个实体类,定义一个属性,编写set/get方法,在属性上用上一步自定义的注解

 1 /**
 2  * 注解使用
 3  */
 4 public class Apple2 {
 5 
 6     @FruitProvider(id = 1, name = "陕西红富士集团", address = "陕西省西安市延安路")
 7     private String appleProvider;
 8 
 9     public void setAppleProvider(String appleProvider) {
10         this.appleProvider = appleProvider;
11     }
12 
13     public String getAppleProvider() {
14         return appleProvider;
15     }
16 
17 }

3.编写注解处理器类,判断一个类的属性上是否有特定注解,如有则打印注解的属性值,用main方法测试注解

 1 /**
 2  * 注解处理器
 3  */
 4 public class FruitInfoUtil {
 5 
 6     public static void getFruitInfo(Class<?> clazz) {
 7         String strFruitProvicer = "供应商信息:\n";
 8         Field[] fields = clazz.getDeclaredFields();//通过反射获取处理注解
 9         for (Field field : fields) {
10             if (field.isAnnotationPresent(FruitProvider.class)) {
11                 FruitProvider fruitProvider = (FruitProvider) field.getAnnotation(FruitProvider.class);
12                 //注解信息的处理地方
13                 strFruitProvicer += "供应商编号:" + fruitProvider.id() + "\n供应商名称:" + fruitProvider.name() + "\n供应商地址:"+ fruitProvider.address();
14                 System.out.println(strFruitProvicer);
15             }
16         }
17     }
18 
19 }
20 
21 class FruitRun{
22     public static void main(String[] args){
23         FruitInfoUtil.getFruitInfo(Apple2.class);
24         /****************输出结果***************/
25         // 供应商编号:1 供应商名称:陕西红富士集团 供应商地址:陕西省西安市延安路
26     }
27 }

posted on 2023-07-17 11:22  德邦总管  阅读(7)  评论(0编辑  收藏  举报

导航