Java 学习之反射机制“解刨”分解类,并获取内容!
正常情况下,单纯的做开发是接触不到反射机制的(额,当然并不排除例外的情况了)。下面我就对我学到的反射方面的知识做一个小小的总结,旨在复习和以后的查看。
原理分析:
所谓反射就是将一个类当做我们研究的对象,进行分解,获取其中的内容的过程。其核心就是Class类,他是所有类的研究的对象,基于Class我们就可以获取一个类的字节码,进而获取想要的值。
目标类(待解刨的类)
package reflect;
public class Person {
public Person(){}
private String name;
public int address;
public int getAddress() {
return address;
}
public void setAddress(int address) {
this.address = address;
}
public void setName(String name) {
this.name = name;
}
public Person(String name ){
this.name = name;
}
public String getName(){
return this.name;
}
private Person(int name){
System.out.println("这里是私有的带有参数的构造方法!");
}
public void eat(){
System.out.println("吃饭!");
}
public void drink(String name){
System.out.println("正在喝 :"+ name);
}
public void purchase(String address ,int [] number , String [] item){
System.out.println("这个人在 " + address +" 买了 "+item +" 分别 "+ number+" 个!");
}
private String flay(String tool){
return "I Can Flay By "+ tool;
}
public static String spend(String money){
System.out.println("我只能够花费 "+money +"钱!");
return "你造吗?";
}
public static void main(String[] args) {
System.out.println(args[0]);
}
}
反射构造方法小例子
package reflect;
import java.lang.reflect.Constructor;
import org.junit.Test;
public class ReflectConstructor {
/**
* 反射得到无参的构造函数的方法
* @throws Exception
*/
@Test
public void reflect1() throws Exception{
Class cls = Class.forName("reflect.Person");
Constructor c = cls.getConstructor(null);
Person person = (Person) c.newInstance(null);
System.out.println("反射得到的结果是:"+ person.toString());
}
/**
* 反射出带有参数的构造方法
* @throws Exception
*/
@Test
public void reflect2() throws Exception {
Class cls = Class.forName("reflect.Person");
Constructor c = cls.getConstructor(String.class);
Person person = (Person) c.newInstance("Name");
System.out.println("反射获得的有参的结果是:"+ person.getName());
}
/**
* 反射出私有的带有参数的构造方法
* @throws NoSuchMethodException
* @throws Exception
*/
@Test
public void reflect3() throws NoSuchMethodException, Exception {
Class cls = Class.forName("reflect.Person");
Constructor c = cls.getDeclaredConstructor(int.class);
//java语言默认私有的方法是不能被外界访问的到的,但要进行强制的访问需要加上一层权限
c.setAccessible(true);
Person person = (Person) c.newInstance(201492115);
System.out.println("反射出私有的带有参数的构造方法!"+ person.toString());
}
}
反射所有方法的小例子
package reflect;
import java.lang.reflect.Method;
import org.junit.Test;
public class ReflectMethods {
/**
* 反射测试一个无参的方法
* public void eat()
* @throws Exception
*/
@Test
public void feflect1() throws Exception {
Class cls = Class.forName("reflect.Person");
Method method =cls.getMethod("eat", null);
System.out.println(method.toString());
method.invoke(new Person(), null);
method.invoke(cls.newInstance(), null);
}
/**
* 反射测试一个带有参数的方法
* public void drink(String name)
* @throws Exception
*/
@Test
public void reflect2() throws Exception {
Class cls = Class.forName("reflect.Person");
Method method =cls.getMethod("drink", String.class);
System.out.println(method.toString());
method.invoke(new Person(), "中国凉茶");
method.invoke(cls.newInstance(), "中国凉茶");
}
/**
* 反射测试含有多个复杂类型的参数的方法
* public void purchase(String address ,int [] number , String [] item)
* @throws Exception
*/
@Test
public void reflect3() throws Exception {
Class cls = Class.forName("reflect.Person");
Method method = cls.getMethod("purchase", String.class,int[].class,String[].class);
method.invoke(cls.newInstance(), "大连",new int[]{1,2,3},new String[]{"鱿鱼","贝壳","螃蟹"});
}
/**
* 反射测试一个带有返回值的不含参数的方法
* public String getName()
* @throws Exception
*/
@Test
public void reflect4() throws Exception {
Class cls = Class.forName("reflect.Person");
Method method = cls.getMethod("getName", null);
String result = (String) method.invoke(cls.newInstance(), null);
System.out.println("测试带有返回值的无参的方法,所得的结果是: "+result);
}
/**
* 反射测试私有含参数的方法
* private String flay(String tool)
*/
@Test
public void reflect5() throws Exception {
Class cls = Class.forName("reflect.Person");
//对于私有的方法只能通过getDeclaredMethod方式得到
Method method = cls.getDeclaredMethod("flay", String.class);
//记得强制将访问人权限提升起来到"可以访问的到"
method.setAccessible(true);
String result = (String) method.invoke(cls.newInstance(), "蝙蝠侠的风衣!");
System.out.println("反射测试私有的含参数的方法的结果是: "+ result);
}
/**
* 反射测试私有的含参数的静态的方法
* public static String spend(String money)
*/
@Test
public void reflect6() throws Exception {
Class cls = Class.forName("reflect.Person");
//对于静态的方法只能通过getDeclaredMethod方式得到
Method method = cls.getDeclaredMethod("spend", String.class);
String result = (String) method.invoke(cls.newInstance(), "100000000000000元人民币!");
System.out.println("反射测试私有的含参数的方法的结果是: "+ result);
}
/**
* 反射测试主方法
* public static void main(String[] args)
* @throws Exception
*/
@Test
public void reflect7() throws Exception {
Class cls = Class.forName("reflect.Person");
Method method = cls.getMethod("main",String[].class);
method.invoke(null, (Object)new String[]{"A","B","C"});
}
}
反射字段值的小例子
package reflect;
import java.lang.reflect.Field;
import org.junit.Test;
public class ReflectField {
/**
* 反射测试字段(属性)的值
* public int address
* @throws Exception
*/
@Test
public void reflect1 () throws Exception {
Class cls = Class.forName("reflect.Person");
Field field = cls.getField("address");
String result = field.getName();
System.out.println("反射所得的字段的结果是:"+result);
}
/**
* 反射测试私有的属性(字段)的结果
* private String name
* @throws Exception
*/
@Test
public void reflect2() throws Exception {
Class cls = Class.forName("reflect.Person");
Field field = cls.getDeclaredField("name");
field.setAccessible(true);
String result = field.getName();
System.out.println("反射测试所得的私有的字段(属性)的值为:"+ result);
}
}
我的总结
- 全部的反射方法均交给JUnit测试通过,并没有发现出错的地方
- 反射是一个比较灵活的方式,具体情况一定要灵活的变通
- 对于私有的变量或者方法,要想访问到其值,就必须打开访问的权限即设置为setAccessable(true);
- 对于框架的运行,需要给出一个类的完整名称(包含完整的包名),这样才不会出错!
Java 学习之反射机制“解刨”分解类,并获取内容!的更多相关文章
- java学习之反射机制
java语言区别于C,C++等准静态语言的最大特点就是java的反射机制.静态语言的最直接定义就是不能在运行时改变程序结构或变量的类型.按照这样的定义,python,ruby是动态语言,C,C++,J ...
- java学习--Reflection反射机制
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意方法和属性:这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制. ...
- Java学习:反射机制简介
反射机制是什么 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法: 对于任意一个对象,都能够调用它的任意一个方法和属性: 这种动态获取的信息以及动态调用对象的方法的功能称为ja ...
- Java学习之反射机制及应用场景
前言: 最近公司正在进行业务组件化进程,其中的路由实现用到了Java的反射机制,既然用到了就想着好好学习总结一下,其实无论是之前的EventBus 2.x版本还是Retrofit.早期的View注解框 ...
- JAVA 学习笔记 - 反射机制
1. JAVA反射机制的概念 2. 怎样实例化一个 Class对象 Class.forName(包名.类名); 对象.getClass(); 类.class; ================== ...
- Java学习笔记-反射机制
Java反射机制实在运行状态时,对于任意一个类,都能够知道这个类的属性和方法,对于任意一个对象,都能够调用他的任意一个属性和方法 获取Class对象的三种方式 Object类中的getClass()方 ...
- Java中的反射机制Reflection
目录 什么是反射? 获取.class字节码文件对象 获取该.class字节码文件对象的详细信息 通过反射机制执行函数 反射链 反射机制是java的一个非常重要的机制,一些著名的应用框架都使用了此机制, ...
- 浅说Java中的反射机制(二)
写过一篇Java中的反射机制,不算是写,应该是抄了,因为那是别人写的,这一篇也是别人写的,摘抄如下: 引自于Java基础--反射机制的知识点梳理,作者醉眼识朦胧.(()为我手记) 什么是反射? 正常编 ...
- 浅说Java中的反射机制(一)
在学习传智播客李勇老师的JDBC系列时,会出现反射的概念,由于又是第一次见,不免感到陌生.所以再次在博客园找到一篇文章,先记录如下: 引用自java中的反射机制,作者bingoideas.(()为我手 ...
随机推荐
- Jenkins简明入门(二) -- 利用Jenkins完成Python程序的build、test、deployment
大家可能还没搞清楚,Jenkins到底能做什么? 本节内容利用Jenkins完成python程序的build.test.deployment,让大家对Jenkins能做的事情有一个直观的了解. 本节内 ...
- 记录一次widora sdk编译ipk 实战编译redis
因为业务需求,需要用到redis存储一点简单的数据,因为redis有良好的哈希机制,可以完美实现我的某些需求,但openwrt官方提供memcached的ipk并没有提供redis,没办法,只能自 ...
- Win2003及2008R2重启自动登录设置方法
在windows系统中,使用最多的可能就是远程操作了,关于远程操作的那些事很多用户还是有些迷茫的.如果win2003系统远程重启后,要重新登录系统才能启用远程功能,这就十分的麻烦,如何才能实现重启后的 ...
- centos7.2中文乱码解决办法
centos7.2 中文乱码解决办法 1.查看安装中文包: 查看系统是否安装中文语言包 (列出所有可用的公共语言环境的名称,包含有zh_CN) # locale -a |grep "zh_C ...
- 关于 form表单 嵌套问题的解决方案
我们经常是这样嵌套的: <form id="formId1" action="" method="post"> //表单1 &l ...
- Redis Error:/var/redis/run/redis_6379.pid exists, process is already running or crashed
命令service Redis start /var/redis/run/redis_6379.pid exists, process is already running or crashed 引起 ...
- 剖析Vue原理&实现双向绑定MVVM
转自:http://www.w3cmark.com/2016/496.html 本文能帮你做什么? 1.了解vue的双向数据绑定原理以及核心代码模块 2.缓解好奇心的同时了解如何实现双向绑定 为了便于 ...
- day05 Servlet 开发和 ServletConfig 与 ServletContext 对象
day05 Servlet 开发和 ServletConfig 与 ServletContext 对象 1. Servlet 开发入门 - hello world 2. Servlet 的调用过程和生 ...
- Java Socket通信代码片
package zhang; import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.IOExcept ...
- Python中的赋值(复制)、浅拷贝、深拷贝之间的区别
1.赋值: 只是复制了新对象的引用,不会开辟新的内存空间. 2.浅拷贝: 创建新对象,其内容是原对象的引用. 浅拷贝有三种形式:切片操作,工厂函数,copy模块中的copy函数. 如: ...