JAVA反射机制的学习(2)

简介:

JAVA语言中的反射机制:

在Java 运行时 环境中,对于任意一个类,能否知道这个类有哪些属性和方法?
对于任意一个对象,能否调用他的方法?这些答案是肯定的,这种动态获取类的信息,以及动态调用类的方法的功能来源于JAVA的反射。从而使java具有动态语言的特性。

JAVA反射机制主要提供了以下功能:

  1.在运行时判断任意一个对象所属的类
  2.在运行时构造任意一个类的对象
  3.在运行时判断任意一个类所具有的成员变量和方法(通过反射甚至可以调用private方法)
  4.在运行时调用任意一个对象的方法(*****注意:前提都是在运行时,而不是在编译时)

Java 反射相关的API简介:

  位于java。lang。reflect包中
    --Class类:代表一个类
    --Filed类:代表类的成员变量
    --Method类:代表类的方法
    --Constructor类:代表类的构造方法
    --Array类:提供了动态创建数组,以及访问数组的元素的静态方法。该类中的所有方法都是静态方法
----Class类
     在 java 的Object类中的申明了数个应该在所有的java类中被改写的methods:

hashCode(), equals(),clone(),toString(),getClass()等,其中的getClass()返回一个
Class 类型的对象。

 Class类十分的特殊,它和一般的类一样继承自Object,其实体用以表达java程序运行

时的 class和 interface,也用来表达 enum,array,primitive,Java Types 以及关键字void
,当加载一个类,或者当加载器(class loader)的defineClass()被JVM调用,便产生一个Class
对象,

 Class是Reflection起源,针对任何你想探勘的class(类),唯有现为他产生一个Class

的对象,接下来才能经由后者唤起为数十多个的反射API。

     Java允许我们从多种途径为一个类class生成对应的Class对象。
          --运用 getClass():Object类中的方法,每个类都拥有此方法 
                                String str="abc";
                                Class cl=str.getClass();

         --运用 Class.getSuperclass():Class类中的方法,返回该Class的父类的Class
         --运用 Class.forName()静态方法:
         --运用 ,Class:类名.class
         --运用primitive wrapper classes的TYPE语法: 基本类型包装类的TYPE,如:Integer.TYPE
                      注意:TYPE的使用,只适合原生(基本)数据类型
----运行时生成instance
 想生成对象的实体,在反射动态机制中有两种方法,一个针对无变量的构造方法,一个针对带参数的

构造方法,如果想调用带参数的构造方法,就比较的麻烦,不能直接调用Class类中的newInstance()
,而是调用Constructor类中newInstance()方法,首先准备一个Class[]作为Constructor的参数类型。
然后调用该Class对象的getConstructor()方法获得一个专属的Constructor的对象,最后再准备一个
Object[]作为Constructor对象昂的newInstance()方法的实参。

  在这里需要说明的是 只有两个类拥有newInstance()方法,分别是Class类和Constructor类

Class类中的newInstance()方法是不带参数的,而Constructro类中的newInstance()方法是带参数的
需要提供必要的参数。

例:
      Class c=Class.forName("DynTest");
      Class[] ptype=new Class[]{double.class,int.class};
      Constructor ctor=c.getConstructor(ptypr);
      Object[] obj=new Object[]{new Double(3.1415),new Integer(123)};
      Object object=ctor.newInstance(obj);
      System.out.println(object);

----运行时调用Method

这个动作首先准备一个Class[]{}作为getMethod(String name,Class[])方法的参数类型,接下来准备一个

Obeject[]放置自变量,然后调用Method对象的invoke(Object obj,Object[])方法。

 注意,在这里调用

----运行时调用Field内容

变更Field不需要参数和自变量,首先调用Class的getField()并指定field名称,获得特定的Field对象后

便可以直接调用Field的 get(Object obj)和set(Object obj,Object value)方法

java 代码
package cn.com.reflection;   
  
import java.lang.reflect.Field;   
import java.lang.reflect.InvocationTargetException;   
import java.lang.reflect.Method;   
  
public class ReflectTester {   
  
    /**  
     * 在这个类里面存在有copy()方法,根据指定的方法的参数去 构造一个新的对象的拷贝  
     * 并将他返回  
     * @throws NoSuchMethodException   
     * @throws InvocationTargetException   
     * @throws IllegalAccessException   
     * @throws InstantiationException   
     * @throws SecurityException   
     * @throws IllegalArgumentException   
     */  
    public Object copy(Object obj) throws IllegalArgumentException, SecurityException, InstantiationException, IllegalAccessException, InvocationTargetException, NoSuchMethodException{   
           
        //获得对象的类型   
        Class classType=obj.getClass();   
        System.out.println("该对象的类型是:"+classType.toString());   
           
        //通过默认构造方法去创建一个新的对象,getConstructor的视其参数决定调用哪个构造方法   
        Object objectCopy=classType.getConstructor(new Class[]{}).newInstance(new Object[]{});   
           
        //获得对象的所有属性   
        Field[] fields=classType.getDeclaredFields();   
           
        for(int i=0;i<fields.length;i++{
            //获取数组中对应的属性   
            Field field=fields[i];   
               
            String fieldName=field.getName();   
            String stringLetter=fieldName.substring(0, 1).toUpperCase();   
               
            //获得相应属性的getXXX和setXXX方法名称   
            String getName="get"+stringLetter+fieldName.substring(1);   
            String setName="set"+stringLetter+fieldName.substring(1);   
               
            //获取相应的方法   
            Method getMethod=classType.getMethod(getName, new Class[]{});   
            Method setMethod=classType.getMethod(setName, new Class[]{field.getType()});   
               
            //调用源对象的getXXX()方法   
            Object value=getMethod.invoke(obj, new Object[]{});   
            System.out.println(fieldName+" :"+value);   
               
            //调用拷贝对象的setXXX()方法   
            setMethod.invoke(objectCopy,new Object[]{value});   
               
               
        }   
           
        return objectCopy;   
           
    }   
       
       
    public static void main(String[] args) throws IllegalArgumentException, SecurityException, InstantiationException, IllegalAccessException, InvocationTargetException, NoSuchMethodException {   
        Customer customer=new Customer();   
        customer.setName("hejianjie");   
        customer.setId(new Long(1234));   
        customer.setAge(19);   
           
        Customer customer2=null;   
        customer2=(Customer)new ReflectTester().copy(customer);   
        System.out.println(customer.getName()+" "+customer2.getAge()+" "+customer2.getId());   
           
        System.out.println(customer);   
        System.out.println(customer2);   
           
  
    }   
  
}   
  
  
class Customer{   
       
    private Long id;   
       
    private String name;   
       
    private int age;   
       
       
    public Customer(){   
           
    }   
  
    public int getAge() {   
        return age;   
    }   
  
  
    public void setAge(int age) {   
        this.age = age;   
    }   
  
  
    public Long getId() {   
        return id;   
    }   
  
  
    public void setId(Long id) {   
        this.id = id;   
    }   
  
  
    public String getName() {   
        return name;   
    }   
  
  
    public void setName(String name) {   
        this.name = name;   
    }   
       
}  

java 代码
package cn.com.reflection;   
  
import java.lang.reflect.Array;   
  
public class ArrayTester1 {   
  
    /**  
     * 此类根据反射来创建  
     * 一个动态的数组   
     */  
    public static void main(String[] args) throws ClassNotFoundException {   
           
        Class classType=Class.forName("java.lang.String");   
           
        Object array= Array.newInstance(classType,10);  //指定数组的类型和大小   
           
         //对索引为5的位置进行赋值   
        Array.set(array, 5, "hello");   
           
        String s=(String)Array.get(array, 5);   
           
        System.out.println(s);   
           
           
        //循环遍历这个动态数组   
        for(int i=0;i<((String[])array).length;i++){   
               
            String str=(String)Array.get(array, i);   
               
            System.out.println(str);   
        }   
  
    }   
  
}   

本文出自seven的测试人生公众号最新内容请见作者的GitHub页:http://qaseven.github.io/

目录
相关文章
|
3月前
|
存储 Oracle Java
java零基础学习者入门课程
本课程为Java零基础入门教程,涵盖环境搭建、变量、运算符、条件循环、数组及面向对象基础,每讲配示例代码与实践建议,助你循序渐进掌握核心知识,轻松迈入Java编程世界。
368 0
|
3月前
|
负载均衡 Java API
grpc-java 架构学习指南
本指南系统解析 grpc-java 架构,涵盖分层设计、核心流程与源码结构,结合实战路径与调试技巧,助你从入门到精通,掌握高性能 RPC 开发精髓。
398 7
|
3月前
|
IDE Java 编译器
java编程最基础学习
Java入门需掌握:环境搭建、基础语法、面向对象、数组集合与异常处理。通过实践编写简单程序,逐步深入学习,打牢编程基础。
262 1
|
4月前
|
Java
Java基础学习day08-作业
本作业涵盖Java中Lambda表达式的应用,包括Runnable与Comparator接口的简化实现、自定义函数式接口NumberProcessor进行加减乘及最大值操作,以及通过IntProcessor处理整数数组,实现遍历、平方和奇偶判断等功能,强化函数式编程实践。
90 5
|
4月前
|
Java API 容器
Java基础学习day08-2
本节讲解Java方法引用与常用API,包括静态、实例、特定类型方法及构造器引用的格式与使用场景,并结合代码示例深入解析。同时介绍String和ArrayList的核心方法及其实际应用。
172 1
|
4月前
|
Java 程序员
Java基础学习day08
本节讲解Java中的代码块(静态与实例)及其作用,深入介绍内部类(成员、静态、局部及匿名)的定义与使用,并引入函数式编程思想,重点阐述Lambda表达式及其在简化匿名内部类中的应用。
164 5
|
4月前
|
Java
Java基础学习day07-作业
本作业包含六个Java编程案例:1)动物类继承与多态;2)加油卡支付系统;3)员工管理类设计;4)学生信息统计接口;5)USB设备控制;6)家电智能控制。综合运用抽象类、接口、继承、多态等面向对象技术,强化Java基础编程能力。
204 3
|
4月前
|
Java
Java基础学习day06-作业
本内容为Java基础学习作业,涵盖两个案例:一是通过Card类及其子类GoldenCard、SilverCard实现加油卡系统,体现封装与继承;二是通过Shape类及子类Circle、Rectangle演示多态与方法重写,强化面向对象编程理解。
108 1
|
4月前
|
设计模式 存储 Java
Java基础学习day07
本节讲解Java中的final关键字、单例设计模式、枚举类、抽象类与接口。涵盖常量定义、单例写法(饿汉式/懒汉式)、枚举特点及应用场景,以及抽象类与接口的使用与区别,助力掌握核心面向对象编程思想。
156 1
|
4月前
|
Java
Java基础学习day05-作业
本文为Java基础学习第五天作业,通过五个案例练习类与对象的定义、构造方法、set/get方法及成员方法的应用。涵盖女友、学生、教师、手机和电影等类的设计与测试,强化面向对象编程基础。
98 2