java反射简解
1、首先一个问题,什么是类,类是不是对象?
我们总是说我们new一个对象出来
那么我们还没有new的时候,建造的那个java类是不是对象呢?
是
它是java.lang.Class的对象
对于反射我们首先要知道的就是什么是Class类
java.lang.Class到底是什么,有啥用。
首先我们要得到Class,我们有三种可以得到的方法,代码如下,c1,c2,c3就是我们需要的Class
//任何一个类都有一个隐含的成员变量叫做calss
Class c1 = Man.class; //通过对象的getClass方法获取class
Class c2 = man1.getClass(); //c1和c2表示了Man的类类型,也就是class type Class c3 = null;
c3 = Class.forName("com.clazz.test.Man");
需要知道的是类的类类型是一样的,都指的是一个,所以c1,c2,c3都是一样的。都是相等的。
那我们现在已经拿到了Class,可以用它来干嘛呢?
我们可以通过类的类类型直接创建一个对象,也就是说,没有new了。
需要注意的是下面的代码中newInstance方法会调用类的无参构造方法
//通过类的类类型创建对象。
Man man2 = (Man) c1.newInstance();
下面是所有的测试代码
package com.clazz.test; /**
* Class类的描述测试
* @author XX
*
*/
public class ClazzDemo1 { public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException
{
//Man是java.lang.Class的对象
//任何一个类都是class类的实例对象 Man man1 = new Man();
//man1是Man类的实例对象 //任何一个类都有一个隐含的成员变量叫做calss
Class c1 = Man.class; //通过对象的getClass方法获取class
Class c2 = man1.getClass(); //c1和c2表示了Man的类类型,也就是class type Class c3 = null;
c3 = Class.forName("com.clazz.test.Man"); //通过类的类类型创建对象。
Man man2 = (Man) c1.newInstance();
}
} class Man{
public Man(){
System.out.println("创建了这个类");
}
}
2、静态加载类和动态加载类
new创建对象是静态加载,在编译时刻就需要加载所有可能使用到的类。
通过动态加载类,可以在运行时刻加载使用到的类
也就是我们的c3使用的方法
package com.clazz.test;
public class DynamicLoadClass {
public static void main(String args[]) throws ClassNotFoundException, InstantiationException, IllegalAccessException
{
//静态加载类,直接new出对象,调用方法,一般都是使用这个,但是一旦没有Student类,编译就会报错
//而且如果需要使用老师对象那么代码需要重新写过很多,要把new对象重新写过,
Student s1 = new Student();
s1.speck();
//通过动态加载类,编译时刻,无论学生还是老师类存不存在,都可以编译通过,只有当运行时刻才进行加载
//好处是如果你需要新增一个类,直接实现IPeople接口,只要forName的名字正确即可,而且这个名字可以
//作为一个参数传进来然后进行修改
Class c1 = Class.forName("com.clazz.test.Student");
IPeople p1 = (IPeople) c1.newInstance();
p1.speck();
}
}
package com.clazz.test;
public class Student implements IPeople{
public void speck()
{
System.out.println("我是学生");
}
}
package com.clazz.test;
public interface IPeople {
public void speck();
}
3通过反射拿到一个类的所有的信息
下面这个就是反射最基本的用法,也是反射最神奇的地方
package com.clazz.util; import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method; import com.clazz.test.Student; /**
* calss类的测试工具类
* @author XX
*
*/
public class ClazzUtil { public static void main(String args[])
{
//通过反射能拿到类的所有方法信息
printAll(new Student());
} /**
* 获取这个对象的信息并打印出来
*/
public static void printAll(Object object)
{
Class c = object.getClass(); //打印该类的名字
System.out.println("类的名字" + c.getName()); //打印该类的所有方法(包含从父类继承来的)(不包含私有的方法)
Method[] ms = c.getMethods();
for (Method method : ms) {
System.out.println("方法:" + method.getName() + " 方法的返回值:" + method.getReturnType().getName()); //获取方法的参数
Class<?>[] par = method.getParameterTypes();
for (Class<?> class1 : par) {
System.out.println("方法的入参:" + class1.getName());
}
} //打印该类的方法(不包含从父类继承来的)包含私有方法
Method[] ms2 = c.getDeclaredMethods();
for (Method method : ms2) {
System.out.println("方法1:" + method.getName() + " 方法的返回值:" + method.getReturnType().getName());
} //获取成员变量(不包含私有变量)
Field[] fs = c.getFields();
for (Field field : fs) {
System.out.println("成员变量: " + field.getName());
} //获取成员变量(包含私有变量)
Field[] fs1 = c.getDeclaredFields();
for (Field field : fs1) {
System.out.println("成员变量: " + field.getName());
} //获取构造方法
Constructor[] con = c.getDeclaredConstructors();
for (Constructor constructor : con) {
System.out.println("构造方法:" + constructor.getName());
}
}
}
package com.clazz.test;
public class Student implements IPeople{
private int a1;
public int a12;
public void speck()
{
System.out.println("我是学生");
}
private void speck2()
{
System.out.println("我是学生");
}
}
4通过反射调用类的方法
这个是反射的应用,通过反射调用一个类的方法
package com.clazz.test; import java.lang.reflect.Method; /**
* @author XX
*
*/
public class ReflectMethod { public static void main(String args[]) throws Exception
{
A a = new A(); Class c = a.getClass(); Method m = c.getMethod("add", int.class,int.class); System.out.println(m.invoke(a, 1,2));
}
} class A{
public int add(int a,int b)
{
return a+b;
}
private int minus(int a,int b)
{
return a-b;
}
}
5利用反射解释ArrayList的泛型
package com.clazz.test; import java.lang.reflect.Method;
import java.util.ArrayList; public class ArrayListRflect { public static void main(String args[]) throws Exception
{
/**
* 下面的代码用于证明,在ArrayList中,泛型只在编译之前有效,编译之后泛型对于list的限制
* 是无效的。利用反射就能绕过编译这个阶段在ArrayList添加本不属于该泛型的值
*/
ArrayList<String> list = new ArrayList<String>(); list.add("哈哈");
//list.add(1);如果原来这么写,编译会报错的 //那么怎么在list放个int的1呢? //获取list的类类型
Class c = list.getClass();
//通过类类型拿到方法
Method m = c.getMethod("add",Object.class);
//通过反射调用方法
m.invoke(list, 1); System.out.println(list);
//结果为:[哈哈, 1] //所以证明了上面的观点,也同时很好的证明了反射多神奇
}
}
java反射简解的更多相关文章
- Java 反射详解 转载
java 反射 定义 功能 示例 概要: Java反射机制详解 | |目录 1反射机制是什么 2反射机制能做什么 3反射机制的相关API ·通过一个对象获得完整的包名和类名 ·实例化Class类对象 ...
- java反射 详解!!!!
java反射(特别通俗易懂) 反射是框架设计的灵魂 (使用的前提条件:必须先得到代表的字节码的Class,Class类用于表示.class文件(字节码)) 一.反射的概述 JAVA反射机制是在运行状态 ...
- java反射详解及说明
首先写一个Person类: package lltse.base.reflectdemo; public class Person { private String name ="张三&qu ...
- 【转载】Java 反射详解
目录 1.什么是反射? 2.反射能做什么? 3.反射的具体实现 4.根据反射获取父类属性 4.反射总结 反射反射,程序员的快乐! 1.什么是反射? Java反射就是在运行状态中,对于任意一个类,都能够 ...
- 《Java基础知识》Java 反射详解
定义 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意方法和属性:这种动态获取信息以及动态调用对象方法的功能称为java语言的反射 ...
- java 反射详解
反射的概念和原理 类字节码文件是在硬盘上存储的,是一个个的.class文件.我们在new一个对象时,JVM会先把字节码文件的信息读出来放到内存中,第二次用时,就不用在加载了,而是直接使用之前缓存的这个 ...
- java反射详解
本篇文章依旧采用小例子来说明,因为我始终觉的,案例驱动是最好的,要不然只看理论的话,看了也不懂,不过建议大家在看完文章之后,在回过头去看看理论,会有更好的理解. 下面开始正文. [案例1]通过一个对象 ...
- java反射详解(转)
本篇文章依旧采用小例子来说明,因为我始终觉的,案例驱动是最好的,要不然只看理论的话,看了也不懂,不过建议大家在看完文章之后,在回过头去看看理论,会有更好的理解. 下面开始正文. [案例1]通过一个对象 ...
- Java反射详解及应用示例
反射是Java中最重要的内容之一,了解反射原理对我们学习各种框架具有很大的帮助 反射的原理: 反射应用示例: import java.lang.reflect.Constructor; import ...
随机推荐
- IE6下a标签失效(背景穿透)
background:fixed url(about:blank);有时候做感应区域的时候在ie6下不给背景就会感应不到,比如说当鼠标移到图片的左半边部分,鼠标手势变成向左的箭 头,比如说有些时候a标 ...
- Spring框架--AOP编程
2 手动实现AOP编程 AOP 面向切面的编程, AOP可以实现"业务代码"与"关注点代码"分离 // 保存一个用户 public void add(User ...
- C语言语法tips(不断更新)
语法虽然琐碎,犯了低级错误代价可就大了,列出一些容易混淆概念,避免以后出错 sizeof和strlen的区别在于:1.sizeof是运算符,语言支持的,strlen可以求malloc出来的字符串的长度 ...
- Compile Time Assertion..
The most seen assertion are during runtime, but this one is at compile time, to give the error more ...
- java URL编程
一.URL编程技术 URL是统一资源定位器的缩写,它是指向互联网“资源”的指针.URL表示了Internet上某个资源的地址.URL支持http,file,ftp等 多种协议.通过URL标识,可以直接 ...
- PMP培训感想
终于等到了C打头的邮件,等了几个星期,悬着的心终于放了.1P4M,基本跟平时模块成绩一致.当初模考时,第一次考了126,还算比较满意,毕竟还没开始苦逼看嘛,悲剧的是不管后面再怎么努力,成绩始终在1 ...
- Redis简介四
一..NET使用Redis 和MongoDB一样,在.NET中使用Redis其实也是使用第三方驱动,官网推荐的是使用ServiceStack.Redis(https://github.com/Serv ...
- js cookie 记住用户名密码
function saveUserInfo(){ if($("#remember").attr('checked')=="checked"){ var user ...
- android应用的优化建议(转载)
首先,这是我在http://www.oschina.net/translate/40-developer-tips-for-android-optimization看到的一片文章,感觉挺有道理的,所以 ...
- 1.Java为什么能跨平台运行?请简述原理
因为它有虚拟机(JVM),JAVA程序不是直接在电脑上运行的,是在虚拟机上进行的,每个系统平台都是有自己的虚拟机(JVM),所以JAVA语言能跨平台. 1, java代码不是直接运行在CPU上,而是运 ...