• 【类反射】类反射原理和获取Class对象的三种方式


    什么是类反射—原理?

    ☆什么是反射

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

    (2)反射(Reflection)是Java程序开发语言的特征之一,它允许运行中的Java程序对自身进行检查, 也称自审,并能直接操作程序的内部属性。例如,使用它能获得Java类中各成员的名称并显示出来。

    (3)Java的这一能力在实际应用中应用得很多,在其它的程序语言中根本就不存在这一特性。例如,Pascal、C或者C++中就没有办法在程序中获得函数定义相关的信息。

    (4)JavaBean是类反射的实际应用之一,它能让一些工具可视化的操作软件组件。这些工具通过类反射动态的载入并取得Java组件(类)的属性。后面学习的各种框架,基本上都会有反射的使用。

    这里写图片描述

    ☆反射引例(HelloWorld、USB)

    反射最大的好处就是解耦

    1. 最简单的类反射:(相当于HelloWorld)

      一个ReflectionHelloWorld类演示:

     1 import java.lang.reflect.Constructor;
     2 import java.lang.reflect.Field;
     3 import java.lang.reflect.Method;
     4 
     5 /**
     6  * 1、类反射使用的步骤:<br>
     7  *      (1)Class c=Class.forName("类全名:包名[.子包名].类名");<br>
     8  *      (2)调用c对象中的成员方法:Field(字段)、Method(方法)、Constructor(构造方法)、Modifier(静态方法和常量)<br>
     9  *      (3)遍历出所有信息<br>
    10  */
    11 public class ReflectionHelloWorld {
    12     public static void main(String[] args) {
    13         try {
    14             Class c=Class.forName("cn.yu.reflect.test.UserModel");
    15             /*
    16              * 提供有关类或接口的单个字段信息,以及对他的动态访问权限
    17              */
    18             Field[] flds=c.getDeclaredFields();
    19             for(Field fild:flds){
    20                 System.out.println(fild);
    21             }
    22             System.out.println("------------------------");
    23             /*
    24              *  提供关于类的单个构造方法的信息以及对它的访问权限。得到的构造方法只能是public类型的。
    25              */
    26             Constructor[] cons=c.getConstructors();
    27             for(Constructor con:cons){
    28                 System.out.println(con);
    29             }
    30             System.out.println("------------------------");
    31             /*
    32              * 返回 Method 对象的一个数组,这些对象反映此 Class 对象表示的类或接口声明的所有方法,
    33              * 包括公共、保护、默认(包)访问和私有方法,但不包括继承的方法。
    34              */
    35             Method[] ms=c.getDeclaredMethods();
    36             for(Method m:ms){
    37                 System.out.println(m);
    38             }
    39             System.out.println("------------------------");
    40         } catch (ClassNotFoundException e) {
    41             e.printStackTrace();
    42         }
    43     }
    44 }

    UserModel类:

     1 import java.io.Serializable;
     2 
     3 /**
     4  * 用户uuid,name
     5  */
     6 interface Intera{
     7     public void aa();
     8 }
     9 public class UserModel implements Serializable{
    10     private static final long serialVersionUID = 1L;
    11     private String uuid;
    12     private String name;
    13     private int type;
    14     private String pwd;
    15 
    16     public UserModel(String uuid, String name, int type, String pwd) {
    17         super();
    18         this.uuid = uuid;
    19         this.name = name;
    20         this.type = type;
    21         this.pwd = pwd;
    22     }
    23 
    24     public UserModel(String uuid, String name, int type) {
    25         super();
    26         this.uuid = uuid;
    27         this.name = name;
    28         this.type = type;
    29     }
    30 
    31     public UserModel(){
    32 
    33     }
    34     public void a(){
    35         System.out.println("我是接口的抽象方法");
    36     }
    37     public String getUuid() {
    38         return uuid;
    39     }
    40     public void setUuid(String uuid) {
    41         this.uuid = uuid;
    42     }
    43     public String getName() {
    44         return name;
    45     }
    46     public void setName(String name) {
    47         this.name = name;
    48     }
    49     public int getType() {
    50         return type;
    51     }
    52     public void setType(int type) {
    53         this.type = type;
    54     }
    55     public String getPwd() {
    56         return pwd;
    57     }
    58     public void setPwd(String pwd) {
    59         this.pwd = pwd;
    60     }
    61     public static long getSerialversionuid() {
    62         return serialVersionUID;
    63     }
    64     @Override
    65     public int hashCode() {
    66         final int prime = 31;
    67         int result = 1;
    68         result = prime * result + ((uuid == null) ? 0 : uuid.hashCode());
    69         return result;
    70     }
    71     @Override
    72     public boolean equals(Object obj) {
    73         if (this == obj)
    74             return true;
    75         if (obj == null)
    76             return false;
    77         if (getClass() != obj.getClass())
    78             return false;
    79         UserModel other = (UserModel) obj;
    80         if (uuid == null) {
    81             if (other.uuid != null)
    82                 return false;
    83         } else if (!uuid.equals(other.uuid))
    84             return false;
    85         return true;
    86     }
    87     @Override
    88     public String toString() {
    89         return "ID:	" + uuid + "  , 	" + name + "	 , " + (type);
    90     }
    91 }

    运行结果:

    private static final long cn.yu.reflect.test.UserModel.serialVersionUID
    private java.lang.String cn.yu.reflect.test.UserModel.uuid
    private java.lang.String cn.yu.reflect.test.UserModel.name
    private int cn.yu.reflect.test.UserModel.type
    private java.lang.String cn.yu.reflect.test.UserModel.pwd
    ------------------------
    public cn.yu.reflect.test.UserModel()
    public cn.yu.reflect.test.UserModel(java.lang.String,java.lang.String,int)
    public cn.yu.reflect.test.UserModel(java.lang.String,java.lang.String,int,java.lang.String)
    ------------------------
    public java.lang.String cn.yu.reflect.test.UserModel.toString()
    public int cn.yu.reflect.test.UserModel.hashCode()
    public boolean cn.yu.reflect.test.UserModel.equals(java.lang.Object)
    public java.lang.String cn.yu.reflect.test.UserModel.getName()
    public int cn.yu.reflect.test.UserModel.getType()
    public void cn.yu.reflect.test.UserModel.setName(java.lang.String)
    public static long cn.yu.reflect.test.UserModel.getSerialversionuid()
    public java.lang.String cn.yu.reflect.test.UserModel.getUuid()
    public void cn.yu.reflect.test.UserModel.setType(int)
    public void cn.yu.reflect.test.UserModel.setPwd(java.lang.String)
    public void cn.yu.reflect.test.UserModel.a()
    public java.lang.String cn.yu.reflect.test.UserModel.getPwd()
    public void cn.yu.reflect.test.UserModel.setUuid(java.lang.String)
    ------------------------

    反射使用的三个步骤

    用于反射的类,如Method,可以在java.lang.reflect包中找到。使用这些类的时候必须要遵循三个步骤:

    1. 第一步:获得你想操作的类的java.lang.Class对象。在运行中的Java程序中,用java.lang.Class类来描述类和接口等。
    2. 第二步:调用诸如getDeclaredMethods的方法,取得该类中定义的所有方法的列表。
    3. 第三步:使用反射的API来操作这些信息。

    如下面这段代码:

    1 Class c = Class.forName("java.lang.String");
    2 Method ms[] = c.getDeclaredMethods();
    3 System.out.println(ms[0].toString());

    它将以文本方式打印出String中定义的第一个方法的原型。

    获取Class对象的三种方式:

    ★ 方式一

    通过对象的getClass方法进行获取。这种方式需要具体的类和该类的对象,以及调用getClass方法。

    ★ 方式二

    任何数据类型(包括基本数据类型)都具备着一个静态的属性class,通过它可直接获取到该类型对应的Class对象。这种方式要使用具体的类,然后调用类中的静态属性class完成,无需调用方法,性能更好。

    ★ 方式三

    通过Class.forName()方法获取。这种方式仅需使用类名,就可以获取该类的Class对象,更有利于扩展。

    代码演示:

     1 import org.junit.Test;
     2 /**
     3  * 1、演示获取Class c对象的三种方法
     4  */
     5 public class ReflectGetClass {
     6 
     7     /**
     8      * 法1:通过对象---对象.getClass()来获取c(一个Class对象)
     9      */
    10     @Test
    11     public void get1(){
    12         Person p=new Person("Jack", 23);
    13         Class c=p.getClass();//来自Object方法
    14     }
    15 
    16     /**
    17      * 法2:通过类(类型)---任何数据类型包括(基本数据类型)都有一个静态的属性class ,他就是c 一个Class对象
    18      */
    19     @Test
    20     public void get2(){
    21         Class c=Person.class;
    22         Class c2=int.class;
    23     }
    24 
    25     /**
    26      * 法3:通过字符串(类全名 )---能够实现解耦:Class.forName(str)
    27      */
    28     @Test
    29     public void get3(){
    30         try {
    31             Class c=Class.forName("cn.hncu.reflect.test.Person");
    32         } catch (ClassNotFoundException e) {
    33             e.printStackTrace();
    34         }
    35     }
    36 
    37 }
    
    
  • 相关阅读:
    NET与Matlab结合 —— 最小二乘法直线拟合(C#)
    C#基础概念二十五问
    C# where子句
    最小二乘法
    蛙蛙推荐:一套.net窗体身份验证方案(解决了防止用户重复登陆,session超时等问题)
    在 ASP.NET 上实现锁定表头、支持滚动的表格的做法
    一个使用泛型的工厂类
    反射技术与设计模式
    ASP.NET实现匿名访问控制
    危险字符过滤的类
  • 原文地址:https://www.cnblogs.com/yuandluck/p/9483932.html
Copyright © 2020-2023  润新知