不是吧,还有人连Java最强大的技术之一:反射还没搞懂?赶紧码住

相信很多人都知道反射可以说是Java中最强大的技术了,它可以做的事情太多太多,很多优秀的开源框架都是通过反射完成的,比如最初的很多注解框架,后来因为java反射影响性能,所以被运行时注解APT替代了,java反射有个开源框架jOOR相信很多人都用过,不过我们还是要学习反射的基础语法,这样才能自己写出优秀的框架,当然这里所讲的反射技术,是学习Android插件化技术、Hook技术等必不可少的!

概述

Java 反射是可以让我们在运行时获取类的方法、属性、父类、接口等类的内部信息的机制。也就是说,反射本质上是一个“反着来”的过程。我们通过new创建一个类的实例时,实际上是由Java虚拟机根据这个类的Class对象在运行时构建出来的,而反射是通过一个类的Class对象来获取它的定义信息,从而我们可以访问到它的属性、方法,知道这个类的父类、实现了哪些接口等信息。

何为反射?

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

简而言之,只要你给我一个.class——类的名字,我就能通过反射获取到类的属性和方法。

反射是很多高级技术的基础,Java 中的注解、动态代理,各种框架注入 Spring 、 MyBatis 等都用到了反射技术。

反射机制能做什么

1. 在运行时判断一个对象所属的类

2. 在运行时可以构造任意一个类的对象

3. 在运行时可以任意判断一个类所包含的成员变量和方法

4. 在运行时可以任意调用一个对象的方法

5. 生成动态代理

 Class类

我们知道使用javac能够将.java文件编译为.class文件,这个.class文件包含了我们对类的原始定义信息(父类、接口、构造器、属性、方法等)。.class文件在运行时会被ClassLoader加载到Java虚拟机(JVM)中,当一个.class文件被加载后,JVM会为之生成一个Class对象,我们在程序中通过new实例化的对象实际上是在运行时根据相应的Class对象构造出来的。确切的说,这个Class对象实际上是java.lang.Class泛型类的一个实例,比如Class对象即为一个封装了MyClass类的定义信息的Class实例。由于java.lang.Class类不存在公有构造器,因此我们不能直接实例化这个类,我们可以通过以下方法获取一个Class对象。

Class 类在 JDK 中的定义

public final class Class<T>

extends Object

implements Serializable, GenericDeclaration, Type, AnnotatedElement

类 Class 的实例表示运行中的 Java 应用程序中的类和接口。

枚举是一种类,注释是一种接口。

每个数组也属于一个类,这个类反映为一个类对象,由具有相同元素类型和维数的所有数组共享。

原始Java类型(布尔型、字节型、char型、short型、int型、long型、float型和double型)和关键字void也被表示为类对象。

类的加载过程

Class类的实例表示正在运行的Java应用程序中的类和接口。每个类只会产生一个Class对象,在类加载的时候自动创建 

获取Class类的三种方式

1 通过 class.forname()来获取Class对象    

        Class clazz = Class.forName("com.mashibing.entity.Emp");

        System.out.println(clazz.getPackage());

        System.out.println(clazz.getName());

        System.out.println(clazz.getSimpleName());

        System.out.println(clazz.getCanonicalName());

2 通过类名.class来获取Class对象

        Class<Emp> clazz = Emp.class;

        System.out.println(clazz.getPackage());

        System.out.println(clazz.getName());

        System.out.println(clazz.getSimpleName());

        System.out.println(clazz.getCanonicalName());

3 通过对象的getClass()来获取Class对象

        Class clazz = new Emp().getClass();

        System.out.println(clazz.getPackage());

        System.out.println(clazz.getName());

        System.out.println(clazz.getSimpleName());

        System.out.println(clazz.getCanonicalName());

4如果是一个基本数据类型,那么可以通过Type的方式来获取Class对象

        Class type = Integer.TYPE;

        System.out.println(type.getName());

        System.out.println(type.getCanonicalName());

反射常用到的API

获取类的构造方法:

public Constructor<T> getConstructor(Class<?>... parameterTypes)

                              throws NoSuchMethodException,

                                    SecurityException

public Constructor<?>[] getConstructors()

                                throws SecurityException


获取类的成员变量

public Field getField(String name)

              throws NoSuchFieldException,

                      SecurityException

public Field[] getFields()

                  throws SecurityException


获取类的方法

public Method getMethod(String name,

                        Class<?>... parameterTypes)

                throws NoSuchMethodException,

                        SecurityException

public Method getMethod(String name,

                        Class<?>... parameterTypes)

                throws NoSuchMethodException,

                        SecurityException

反射在 Spring 中的应用举例

反射在众多框架中都有普遍的应用。比如Spring IOC容器帮我们实例化众多的bean,下面我们简单模拟一下反射在其中起到的作用。

此处使用的案例接这篇:【设计模式】代理模式那些事儿:静态代理,动态代理,JDK的动态代理,cglib,Spring AOP

Spring 配置文件:

<bean id="pony" class="com.xblzer.dp.proxy.springaop.Pony"></bean>


使用的时候直接这样就能拿到定义的类了:

ApplicationContext ctx = new ClassPathXmlApplicationContext("app_aop.xml");

Pony pony = (Pony) ctx.getBean("pony");


那么是怎么做到的呢?就是通过反射。

Spring 通过配置文件实例化对象,并将其放到容器的过程大概就是(模拟):

//伪代码

//1.解析<bean .../>元素的id属性得到该字符串值为“pony” 

String idStr = "pony"; 

//解析<bean .../>元素的class属性得到该字符串值为“com.xblzer.dp.proxy.springaop.Pony” 

String classStr = "com.xblzer.dp.proxy.springaop.Pony"; 

//利用反射机制,通过classStr获取Class类对象 

Class<?> cls = Class.forName(classStr); 

//实例化对象 

Object obj = cls.newInstance(); 

//放到Spring容器 

Map<String, Object> container = new HashMap<>();

container.put(idStr, obj);

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容

  • 反射和类对象 当一个类被java虚拟机加载,那么它会创建一个类对象。每一个类都有并且只有一个类对象。类对象保 存了...
    在远方的你等我阅读 362评论 1 3
  • 概述很多主流的IOC框架,像移动端的Retrofit、服务端的Spring等的核心思想都是通过反射去实现的,阅读源...
    Horps阅读 373评论 0 1
  • 类加载器当程序要使用某个类时,如果该类还没有被加载到内存中,则系统会通过加载,链接,初始化这三步来实现对这个类进行...
    Stringer阅读 313评论 0 0
  • 反射授课目录: ###01_反射(类的加载概述和加载时机) * A:类的加载概述 * 当程序要使用某个类时,如果该...
    猿学阅读 232评论 0 0
  • 注解 一、前期概要 1、 什么是反射 Java 反射机制在程序运行时,对于任意一个类,都能够知道这个类的所有属性和...
    唯老阅读 311评论 0 4