• 反射:调用运行时类的指定结构


     

     

     

    package com.atguigu.java2;
    
    import com.atguigu.java1.Person;
    import org.junit.Test;
    
    import java.lang.reflect.Constructor;
    import java.lang.reflect.Field;
    import java.lang.reflect.Method;
    
    /**
     * 调用运行时类中指定的结构:属性、方法、构造器
     *
     * @author CH
     * @create 2021 下午 4:46
     */
    public class ReflectionTest {
    
        /*
    
            不需要掌握
         */
        @Test
        public void testField() throws Exception {
            Class clazz = Person.class;
    
            //创建运行时类的对象
            Person p = (Person) clazz.newInstance();
    
    
            //获取指定的属性:要求运行时类中属性声明为public
            //通常不采用此方法
            Field id = clazz.getField("id");
    
            /*
            设置当前属性的值
    
            set():参数1:指明设置哪个对象的属性   参数2:将此属性值设置为多少
             */
    
            id.set(p,1001);
    
            /*
            获取当前属性的值
            get():参数1:获取哪个对象的当前属性值
             */
            int pId = (int) id.get(p);
            System.out.println(pId);
    
    
        }
        /*
        如何操作运行时类中的指定的属性 -- 需要掌握
         */
        @Test
        public void testField1() throws Exception {
            Class clazz = Person.class;
    
            //创建运行时类的对象
            Person p = (Person) clazz.newInstance();
    
            //1. getDeclaredField(String fieldName):获取运行时类中指定变量名的属性
            Field name = clazz.getDeclaredField("name");
    
            //2.保证当前属性是可访问的
            name.setAccessible(true);
            //3.获取、设置指定对象的此属性值
            name.set(p,"Tom");
    
            System.out.println(name.get(p));
        }
    
        /*
        如何操作运行时类中的指定的方法 -- 需要掌握
         */
        @Test
        public void testMethod() throws Exception {
    
            Class clazz = Person.class;
    
            //创建运行时类的对象
            Person p = (Person) clazz.newInstance();
    
            /*
            1.获取指定的某个方法
            getDeclaredMethod():参数1 :指明获取的方法的名称  参数2:指明获取的方法的形参列表
             */
            Method show = clazz.getDeclaredMethod("show", String.class);
            //2.保证当前方法是可访问的
            show.setAccessible(true);
    
            /*
            3. 调用方法的invoke():参数1:方法的调用者  参数2:给方法形参赋值的实参
            invoke()的返回值即为对应类中调用的方法的返回值。
             */
            Object returnValue = show.invoke(p,"CHN"); //String nation = p.show("CHN");
            System.out.println(returnValue);
    
            System.out.println("*************如何调用静态方法*****************");
    
            // private static void showDesc()
    
            Method showDesc = clazz.getDeclaredMethod("showDesc");
            showDesc.setAccessible(true);
            //如果调用的运行时类中的方法没有返回值,则此invoke()返回null
    //        Object returnVal = showDesc.invoke(null);
            Object returnVal = showDesc.invoke(Person.class);
            System.out.println(returnVal);//null
    
        }
    
        /*
        如何调用运行时类中的指定的构造器
         */
        @Test
        public void testConstructor() throws Exception {
            Class clazz = Person.class;
    
            //private Person(String name)
            /*
            1.获取指定的构造器
            getDeclaredConstructor():参数:指明构造器的参数列表
             */
    
            Constructor constructor = clazz.getDeclaredConstructor(String.class);
    
            //2.保证此构造器是可访问的
            constructor.setAccessible(true);
    
            //3.调用此构造器创建运行时类的对象
            Person per = (Person) constructor.newInstance("Tom");
            System.out.println(per);
    
        }
    
    }
    @Test
        public void test1(){
    
            Class clazz = Person.class;
    
            //获取属性结构
            //getFields():获取当前运行时类及其父类中声明为public访问权限的属性
            Field[] fields = clazz.getFields();
            for(Field f : fields){
                System.out.println(f);
            }
            System.out.println();
    
            //getDeclaredFields():获取当前运行时类中声明的所有属性。(不包含父类中声明的属性)
            Field[] declaredFields = clazz.getDeclaredFields();
            for(Field f : declaredFields){
                System.out.println(f);
            }
        }

    不积跬步,无以至千里;不积小流,无以成江海。
  • 相关阅读:
    23种设计模式(转载)
    RabbitMQ JAVA客户端调用
    JavaScript中的this
    RedisDesktopManager 踩坑之旅
    webmagic使用手册
    Maven 手动添加本地jar包
    根据端口号查询 进程 并杀掉进程
    从经典面试题看java中类的加载机制
    Java线程的5种状态及切换(透彻讲解)
    JVM 类加载机制详解
  • 原文地址:https://www.cnblogs.com/CCTVCHCH/p/14928572.html
Copyright © 2020-2023  润新知