Java反射机制概念及使用
反射机制 —— 将类中的所有成员反射成对于的类。
|
|
转换对应的类 |
获取方法 |
说明 |
|
类 |
Class |
Class mClass = Person.class; | 通过类名直接获取 |
| Class mClass = new Person().getClass(); | 通过对象获取 | ||
| Class mClass = Class.forName("com.test.Person"); | 通过全类名获取 | ||
| Class superClass = mClass.getSuperclass(); | 获取父类 | ||
|
成员变量 |
Field |
Field[] fields = clazz.getDeclaredFields(); | 获取全部成员变量的Field |
| Field field1 = mClass.getDeclaredField("name"); | 获指定变量的 Field | ||
| Object val = field1.get(new Person()); | 获取指定变量的 Field 的值 | ||
| field1.set(new Person(), "张三"); | 设置指定变量的 Field 的值 | ||
|
构造方法 |
Constructor |
Class<Person> clazz = (Class<Person>) Class.forName(className); | 获取指定类的 Class |
| Constructor<Person>[] constructors = (Constructor<Person>[]) Class.forName(className).getConstructors(); | 获取全部构造方法的 Constructor | ||
| Constructor<Person> constructor = clazz.getConstructor(String.class, int.class); | 获取指定的构造方法的 Constructor (参数代表构造方法的参数类型) | ||
|
成员方法 |
Method |
Method[] methods = mClass.getMethods(); | 获取部分成员方法(不能获取private方法) |
| Method[] methods = clazz.getDeclaredMethods(); | 获取全部成员方法 | ||
|
Method method1 = clazz.getDeclaredMethod("setName", String.class); Method method2 = clazz.getDeclaredMethod("setName", String.class, int.class); |
获取指定方法(参1:方法名,参2之后:方法参数类型) |

package steffen.reflex;
public class Persion {
public String name="李四";
private int age = 10;
public Persion(){}
public Persion(String name, int age){
this.name = name;
this.age = age;
}
public String getName(){return name;}
private int getAge(){return age;}
}
/**
* Class 类对象
* @throws ClassNotFoundException
*/
private void classTest() throws ClassNotFoundException{
Class classString = String.class;
Class classString2 = new String().getClass();
Class classString3 = Class.forName("java.lang.String"); // 不支持内部类 System.out.println("1.1 classString="+classString);
System.out.println("1.2 classString2="+classString2);
System.out.println("1.3 classString3="+classString3); // 总结: Class对象有系统自动创建,且只创建一次。
}
// 输出结果
1.1 classString=class java.lang.String
1.2 classString2=class java.lang.String
1.3 classString3=class java.lang.String
constructorTest("steffen.reflex.Persion");
/**
* constructor 构造方法对象
* @param className
* @throws ClassNotFoundException
*/
private void constructorTest(String className) throws ClassNotFoundException {
Class<?> clazz = Class.forName(className);
Constructor<?>[] constructors = (Constructor<?>[])clazz.getConstructors();
// 1.获取全部构造方法
System.out.println("2.1 获取"+className+"全部构造方法:");
for (Constructor<?> element : constructors) {
System.out.println(element+"; ");
}
System.out.println();
// 2.获取某个构造方法
Constructor<?> constructors2;
try {
constructors2 = clazz.getConstructor(String.class, int.class);
System.out.println("2.2 获取某个特定的构造方法:"+constructors2);
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
}
}
// 输出结果
2.1 获取steffen.reflex.Persion全部构造方法:
public steffen.reflex.Persion();
public steffen.reflex.Persion(java.lang.String,int); 2.2 获取某个特定的构造方法:public steffen.reflex.Persion(java.lang.String,int)
fieldTest("steffen.reflex.Persion");
/**
* Field 变量对象
* @param className
* @throws ClassNotFoundException
*/
private void fieldTest(String className) throws ClassNotFoundException{
Class<?> clazz = Class.forName(className);
// 获取所以变量
Field[] arrField = clazz.getDeclaredFields();
System.out.println("3.1 显示"+className+"所有变量:");
for (Field field : arrField) {
System.out.println(field+"; ");
}
System.out.println();
// 获取某个特定的变量
try {
Persion mPersion = new Persion();
Field fieldName = clazz.getDeclaredField("name");
System.out.println("3.2 获取名为name的变量对象:"+fieldName);
// 设置该对象的值:
try {
fieldName.set(mPersion, "张三");
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
// 获取该变量的值
try {
Object obj = fieldName.get(mPersion);
System.out.println("3.3 名为name的值为:"+obj);
} catch (IllegalArgumentException e1) {
e1.printStackTrace();
} catch (IllegalAccessException e1) {
e1.printStackTrace();
}
} catch (NoSuchFieldException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
}
}
// 输出结果
3.1 显示steffen.reflex.Persion所有变量:
public java.lang.String steffen.reflex.Persion.name;
private int steffen.reflex.Persion.age; 3.2 获取名为name的变量对象:public java.lang.String steffen.reflex.Persion.name
3.3 名为name的值为:张三
methodTest("steffen.reflex.Persion");
/**
* Method 方法对象
* @param className
* @throws ClassNotFoundException
*/
private void methodTest(String className) throws ClassNotFoundException {
Class<?> clazz = Class.forName(className);
// 5.1 获取类所有的方法
Method[] arrMethod = clazz.getDeclaredMethods();
System.out.println("4.1 获取类的所有方法:");
for (Method method : arrMethod) {
System.out.println(method+"; ");
}
System.out.println();
// 5.2 获取类的公开方法
Method[] arrMethod2 = clazz.getMethods();
System.out.print("4.2 获取类的所有Public方法:");
for (Method method : arrMethod2) {
System.out.println(method+"; ");
}
System.out.println();
try {
// 获取指定方法
Method methodGetAge = clazz.getDeclaredMethod("getAge");
System.out.println("4.3 获取指定方法:"+methodGetAge);
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
}
}
// 输出结果
4.1 获取类的所有方法:
public java.lang.String steffen.reflex.Persion.getName();
private int steffen.reflex.Persion.getAge(); 4.2 获取类的所有Public方法:public java.lang.String steffen.reflex.Persion.getName();
public final void java.lang.Object.wait() throws java.lang.InterruptedException;
public final void java.lang.Object.wait(long,int) throws java.lang.InterruptedException;
public final native void java.lang.Object.wait(long) throws java.lang.InterruptedException;
public boolean java.lang.Object.equals(java.lang.Object);
public java.lang.String java.lang.Object.toString();
public native int java.lang.Object.hashCode();
public final native java.lang.Class java.lang.Object.getClass();
public final native void java.lang.Object.notify();
public final native void java.lang.Object.notifyAll(); 4.3 获取指定方法:private int steffen.reflex.Persion.getAge()
适用场景及利弊
一、反射的适用场景是什么?
1)Java 的反射机制在做基础框架的时候非常有用,有一句话这么说来着:反射机制是很多 Java 框架的基石。而一般应用层面很少用,不过这种东西,现在很多开源框架基本都已经给你封装好了,自己基本用不着写。典型的除了 Hibernate 之外,还有 Spring 也用到很多反射机制。经典的就是在xml文件或者 properties 里面写好了配置,然后在 Java 类里面解析 xml 或 properties 里面的内容,得到一个字符串,然后用反射机制,根据这个字符串获得某个类的Class实例,这样就可以动态配置一些东西,不用每一次都要在代码里面去 new 或者做其他的事情,以后要改的话直接改配置文件,代码维护起来就很方便了,同时有时候要适应某些需求,Java类里面不一定能直接调用另外的方法,这时候也可以通过反射机制来实现。
总的来说,自己写的很少,具体什么时候要用那要看需求,反射机制无非就是根据一个 String 来得到你要的实体对象,然后调用它原来的东西。但是如果是要自己写框架的话,那就会用得比较多了。
2)当你做一个软件可以安装插件的功能,你连插件的类型名称都不知道,你怎么实例化这个对象呢?因为程序是支持插件的(第三方的),在开发的时候并不知道 。所以无法在代码中 new 出来 ,但反射可以,通过反射,动态加载程序集,然后读出类,检查标记之后再实例化对象,就可以获得正确的类实例。
3)在编码阶段不知道那个类名,要在运行期从配置文件读取类名,这时候就没有办法硬编码 new ClassName(),而必须用到反射才能创建这个对象.反射的目的就是为了扩展未知的应用。比如你写了一个程序,这个程序定义了一些接口,只要实现了这些接口的dll都可以作为插件来插入到这个程序中。那么怎么实现呢?就可以通过反射来实现。就是把 dll 加载进内存,然后通过反射的方式来调用 dll 中的方法。很多工厂模式就是使用的反射。
二、程序员在自己的业务开发中应该尽量的远离反射
反射:在流行的库如 Spring 和 Hibernate 中,反射自然有其用武之地。不过内省业务代码在很多时候都不是一件好事,原因有很多,一般情况下建议不要使用反射。
首先是代码可读性与工具支持。打开熟悉的 IDE,寻找你的 Java 代码的内部依赖,很容易吧。现在,使用反射来替换掉你的代码然后再试一下,结果如何呢?如果通过反射来修改已经封装好的对象状态,那么结果将会变得更加不可控。比如下面这段代码:
Class<?> clazz = Class.forName("steffen.reflex.Persion");
Field[] arrField = clazz.getDeclaredFields();
如果这样做就无法得到编译期的安全保证。就像上面这个示例一样,你会发现如果 getDeclaredField() 方法调用的参数输错了,那么只有在运行期才能发现。要知道的是,寻找运行期 Bug 的难度要远远超过编译期的 Bug。
最后还要谈谈代价问题。JIT 对反射的优化程度是不同的,有些优化时间会更长一些,而有些甚至是无法应用优化。因此,有时反射的性能损失可以达到几个数量级的差别。不过在典型的业务应用中,你可能不会注意到这个代价。
总结一下,我觉得在业务代码中唯一合理(直接)使用反射的场景是通过 AOP。除此之外,你最好远离反射这一特性。
三、性能分析
反射机制是一种程序自我分析的能力。用于获取一个类的类变量,构造函数,方法,修饰符。
优点:运行期类型的判断,动态类加载,动态代理使用反射。
缺点:性能是一个问题,反射相当于一系列解释操作,通知 jvm 要做的事情,性能比直接的 java 代码要慢很多。
Java反射机制概念及使用的更多相关文章
- Java反射机制概念及应用场景
Java的反射机制相信大家在平时的业务开发过程中应该很少使用到,但是在一些基础框架的搭建上应用非常广泛,今天简单的总结学习一下. 1. 什么是反射机制? Java反射机制是在运行状态中,对于任意一个类 ...
- Java反射机制的基本概念与使用
本篇文章分为以下几个部分: 1.认识反射 2.反射的源头(Class类) 3.利用反射操作构造方法 4.利用反射调用类中的方法 5.反射中的invoke方法 6.利用反射调用类中的属性 反射在我们普通 ...
- Java反射机制一 概念和简单的使用方法。
一 概念 java反射机制属于 java动态性之一 ,指的是可以运行时加载,探知,使用编译期间完全未知的类,程序在运行状态中,可以动态的加载一个只有, 名称的类,对于任意一个已加载的类,都能够知道这 ...
- 第28章 java反射机制
java反射机制 1.类加载机制 1.1.jvm和类 运行Java程序:java 带有main方法的类名 之后java会启动jvm,并加载字节码(字节码就是一个类在内存空间的状态) 当调用java命令 ...
- java反射机制深入详解
java反射机制深入详解 转自:http://www.cnblogs.com/hxsyl/archive/2013/03/23/2977593.html 一.概念 反射就是把Java的各种成分映射成 ...
- 反射——Java反射机制
反射概述 什么是反射? ① 反射的概念是由Smith在1982年首次提出的,主要指程序可以访问.检测和修改它本身状态或行为的一种能力. ② JAVA反射机制是在运行状态中,对应任意一个类,都能 ...
- Java反射机制及IoC原理
一. 反射机制概念 主要是指程序可以访问,检测和修改它本身状态或行为的一种能力,并能根据自身行为的状态和结果,调整或修改应用所描述行为的状态和相关的语义.在java中,只要给定类的名字, 那么就可以通 ...
- [转]java反射机制
原文地址:http://www.cnblogs.com/jqyp/archive/2012/03/29/2423112.html 一.什么是反射机制 简单的来说,反射机制指的是程序在运 ...
- 11.Java反射机制 哦对了,前面的序号来自随笔关于编程之路的思索第一篇
基本概念 在Java运行时环境中,对于任意一个类,能否知道这个类有哪些属性和方法?对于任意一个对象,能否调用它的任意一个方法? 答案是肯定的. 这种动态获取类的信息以及动态调用对象的方法的功能来自于J ...
随机推荐
- 从 Socket 编程谈谈 IO 模型(三)
快过年啦,估计很多朋友已在摸鱼的路上.而我为了兄弟们年后的追逐,却在苦苦寻觅.规划,导致文章更新晚了些,各位猿粉谅解. 上期分享,我们结合新春送祝福的场景,通过一坨坨的代码让 BIO.NIO 编程过程 ...
- hello world: 我的博客写作思路
1. 本人计算机专业,研究生刚毕业,即将入职金融科技领域,决定借博客园平台写自己的博客,原因如下: 从小白到大白,离不开各大学习平台和技术博客的指导和分享,是时候回馈了. 借此机会整理自己从本科.研究 ...
- 判断移动端还是PC端JS
if(/Android|webOS|iPhone|iPod|BlackBerry/i.test(navigator.userAgent)) { // alert("手机&qu ...
- jQuery实现回车键抬起触发事件
$(function(){ //回车键按下触发 $(document).keydown(function(event){ if(event.keyCode==13){ alert("niha ...
- java中封装,继承,多态,接口学习总结
### 一:封装java中封装是指一种将抽象性函式接口的实现细节部分包装.隐藏起来的方法. 封装可以被认为是一个保护屏障,防止该类的代码和数据被外部类定义的代码随机访问.要访问该类的代码和数据,必须通 ...
- 原来rollup这么简单之插件篇
大家好,我是小雨小雨,致力于分享有趣的.实用的技术文章. 内容分为翻译和原创,如果有问题,欢迎随时评论或私信,希望和大家一起进步. 大家的支持是我创作的动力. 计划 rollup系列打算一章一章的放出 ...
- 使用 python 创建&更改 word 文档
使用 python 修改 word 文档 说明:这个需求是老师想要一个自动识别 word 文档中指定位置的分数,并填入相应表格. 使用库 python-docx 的官方文档地址是:python-doc ...
- 家庭版记账本app进度之关于listview显示账单,并为其添加点击事件
这个主要学习是关于listview的学习. 怎样去自定义adapter,以及使用.自己创建文件,还有就是为listview的每一个子控件添加点击事件. 在整个过程中收获到的知识点如下: 一.对于数据库 ...
- Unity Shader and Effects Cookbook问题记录
1.p61的specular计算,涉及到的一个参数“_SpecColor”是在Unity的官方cginc文件(UnityLightingCommon.cginc)中,是直接赋颜色给这个参数,反应到你模 ...
- Elasticsearch系列---聚合查询原理
概要 本篇主要介绍聚合查询的内部原理,正排索引是如何建立的和优化的,fielddata的使用,最后简单介绍了聚合分析时如何选用深度优先和广度优先. 正排索引 聚合查询的内部原理是什么,Elastich ...