JAVA反射

反射是Java 语言的特征之一,它允许运行中的Java程序获取自身的信息,并且可以操作类或对象的内部属性。通过反射,可以在运行时获得程序中每一个类型的成员和成员的信息。程序中一般的对象的类型都是在编译期就确定下来的,而Java反射机制可以动态地创建对象并调用其属性,这样的对象的类型在编译期是未知的。

反射的核心是JVM在运行时才动态加载类或调用方法/访问属性,它不需要事先知道运行对象是谁。

Java反射框架主要提供以下功能:

  1. 在运行时判断任意一个对象所属的类;
  2. 在运行时构造任意一个类的对象;
  3. 在运行时判断任意一个类所具有的成员变量和方法;
  4. 在运行时调用任意一个对象的方法

当我们在使用IDE(如Eclipse,IDEA)时,当我们输入一个对象或类并想调用它的属性或方法时,一按点号,编译器就会自动列出它的属性或方法,这里就会用到反射。

反射最重要的用途就是开发各种通用框架。

操作反射常用的类:Constructor、Field、Method、Array位于java.lang.reflect包中。
通过反射获取类的信息分为两步:

  1. 获取Class对象

    • 调用对象的getClass()方法
    • 调用类的class属性
    • 调用Class类的forName()方法
  2. 通过Class对象获取信息

    • 访问Class对应的类的构造方法
    • 访问Class对应的类的方法
    • 访问Class对应的类的属性

访问Class对应的类的构造方法

访问Class对应的类的构造方法 说明
Constructor getConstructor(Class[ ] params) 返回此Class对象所包含的类的指定的public构造方法
Constructor[ ] getConstructors( ) 返回此Class对象所包含的类的所有的public构造方法
Constructor getDeclaredConstructor(Class[ ] params) 返回此Class对象所包含的类的声明的指定的构造方法
Constructor[ ] getDeclaredConstructors( ) 返回此Class对象所包含的类的声明的所有构造方法

访问Class对应的类的方法

访问Class对应的类的方法 说明
Method getMethod(String name, Class[ ] params) 返回此Class对象所包含的类的指定的public方法
Method[ ] getMethods( ) 返回此Class对象所包含的类的所有的public方法
Method getDeclaredMethod(String name, Class[ ] params) 返回此Class对象所包含的类的声明的指定的方法
Method[ ] getDeclaredMethods( ) 返回此Class对象所包含的类的声明的所有方法

访问Class对应的类的属性

访问Class对应的类的属性 说明
Field getField(String name) 返回此Class对象所包含的类的指定的public属性
Field[ ] getFields( ) 返回此Class对象所包含的类的所有的public属性
Field getDeclaredField(String name) 返回此Class对象所包含的类的声明的指定的属性
Field[ ] getDeclaredFields( ) 返回此Class对象所包含的类的声明的所有属性

Array类操作数组

Array类里定义了大量静态方法,通过类名直接调用。
这里的Array和集合框架里的Arrays不一样,他们的相同点都是定义了大量静态方法,都是数组的操作,Array位于java.lang.relect包,是动态创建并操作数组;Arrays位于java.util包,是一个工具类,其中包含了一些方法可直接实现数组的排序、搜索等。

下面结合代码做下说明:

public class Demo {
          //调用class对象会抛出大量异常,这里为简化代码,直接抛给jvm
    public static void main(String[] args) throws Exception {
        Student stu = new Student("张三",23);
        
        //获取Class对象
        //方法一:forName()方法,必须要完整的包名加类名
        Class<?> stuClass = Class.forName("reflect.Student");
        //方法二:类的class属性
        Class<?> cla1 = Student.class;
        //方法三:getClass()方法
        Class<?> cla2 = stu.getClass();
        System.out.println(stuClass.getName());
        
        //获取声明的所有构造方法
        Constructor<?>[] constructor = stuClass.getConstructors();
        for (int i = 0; i < constructor.length; i++) {
            System.out.println(constructor[i].toString());
        }
        //获取指定的一个有参构造方法
        Constructor<?> con1 = stuClass.getDeclaredConstructor(String.class,int.class);
        Object obj = con1.newInstance("tom",20);  //使用Class对象的newInstance()方法创建对象
        ((Student)obj).shwoInfo();
        
        //获取声明的所有属性
        Field[] fields = stuClass.getDeclaredFields();
        for (int i = 0; i < fields.length; i++) {
            System.out.println(fields[i].toString());
        }
        //获取public name属性,并改变其值
        Field fieldname = stuClass.getDeclaredField("name");
        fieldname.set(stu, "王小二");//set方法给属性赋值
        String nameStr = (String) fieldname.get(stu); //get方法取出属性的值
        System.out.println(nameStr);
        //获取private age属性,并改变其值
        Field fieldAge = stuClass.getDeclaredField("age");
        fieldAge.setAccessible(true); //改变属性访问权限
        fieldAge.set(stu, 58);
        int ageInt = fieldAge.getInt(stu);
        System.out.println(ageInt);
        
        //获得声明的所有方法
        Method[] methods = stuClass.getDeclaredMethods();
        for (int i = 0; i < methods.length; i++) {
            System.out.println(methods[i].toString());//以字符串形式打印出方法名
        }
        //获得声明的指定方法,传值并调用
        Method m1 = stuClass.getDeclaredMethod("sayHelloTo", String.class);
        m1.invoke(null, "jerry");//这里的null是因为调用的方法是静态方法,不用传对象
        //获得其中一个方法,直接调用
        Method m2 = stuClass.getDeclaredMethod("shwoInfo");
        m2.invoke(stu);//用invoke()调用方法时需要传一个对象

        //Array操作数组
        Object arr = Array.newInstance(int.class, 3);//创建元素为3的int数组
        for (int i = 0; i < Array.getLength(arr); i++) {
            Array.set(arr, i, 2*i); //循环给元素赋值
        }
        for (int i = 0; i < Array.getLength(arr); i++) {
            System.out.println(Array.get(arr, i));//循环取值
        }
        Array.set(array, 0, 99);//指定元素赋值
        System.out.println(Array.get(array, 0));//指定元素取值
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容