最近收到很多关于Java反射机制的问题留言,其实Java反射机制技术方面没有太多难点,或许是大家在学习过程中遗漏了细小知识点,导致一些问题无法彻底理解,现在我们简单的总结一下,加深印象。

什么是反射机制?

“JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。”
我们通过一些例子,更好理解反射机制。
Class类

我们知道Java是一门面向对象语言,在Java的世界里,万物皆为对象,比如我们有一个Person类:

public class Person { }

我们创建一个Person类的实例,Person person = new Person; 那么这个person,就是Person类的实例对象。
那么既然万物都为对象,所以类也是对象。

类是什么的对象呢?类是Class类的对象,表示方式有三种:

//第一种,任何一个类都有一个隐含的静态成员变量class Class c1 = Person.class; //第二种,已经知道该类的对象,通过getClass获得 Class c2 = person.getClass; //第三种,Class类的forName方法 Class c3 = Class.forName("Person"); //这里,c1,c2,c3都是Class类的实例,我们称c1, c2 ,c3为Person类的类类型 //不难看出,c1 == c2结果是true, c2 == c3结果也是true
通过类的类类型,我们经常会用到的方法就是newInstance方法,通过该方法可以创建该类的实例:

Person personA = new Person; //直接new一个实例 Person personB = Person.class.newInstance; //通过newInstance方法获得Person的实例 //在学习JAVAEE时候,newInstance方法我们最常见于获取数据库驱动 Class.forName("com.mysql.jdbc.Driver").newInstance; //需要注意的是,在使用newInstance方法的前提是该类必须要有无参构造方法

动态加载类:

编译时刻加载类称为静态加载,运行时刻加载类称为动态加载,使用new方法新建实例即为静态加载类,在编译时候就要加载全部类。这里我们举一个例子:
为了更好的区分编译和运行,我们不适用IDE工具,而使用记事本来实现这个例子:
//我们举女娲造人的例子,创建一个CreatePerson类 public class CreatePerson { public static void main(String[] args) { if(args[0].equalsIgnoreCase("man")) { //如果从命令行传入的参数为man 则创建Man的实例并调用say方法 new Man.say; } if(args[0].equalsIgnoreCase("woman")) { //如果从命令行传入的参数为woman 则创建Woman的实例并调用say方法 new Woman.say; } } } //CreatePerson类用到了2个类,分别是Man和Woman。 //但是我们现在只创建Man类 public class Man { public void say { System.out.println("I am a man !"); } }
我们在CMD中编译CreatePerson,看看会发生什么:

JAVA反射机制—学习总结
提示我们找不到Woman这个类。这就是静态加载,在编译时刻需要加载全部类。那么问题来了,如果我要写一个程序,里面有100个功能,这100个功能分别由100个类实现,那么一旦缺少一个类,这整个程序是不是就不能用了。

为了解决这个问题,我们可以使用动态加载。我们把上面这个例子改一下:

//创建一个Person接口 public interface Person { void say; } //修改Man类,继承Person接口 public class Man implements Person{ public void say { System.out.println("I am a man !"); } } //修改CreatePerson类,实现动态加载类 public class CreatePerson{ public static void main(String[] args) { try{ //动态加载类 Class c = Class.forName(args[0]); //通过类的类类型,创建该类实例对象 Person person = (Person)c.newInstance; person.say; } catch(Exception e) { e.printStackTrace; } } }

这个时候,我们再进行编译:
没有报错了,那么我们继续把Man编译,再运行一下:
如果我们在命令行中输入的参数是Woman呢?
JAVA反射机制—学习总结
抛出ClassNotFoundException异常。因为我们并没有创建Woman类,所以如果以后需要创建Woman类实例,我们只需要新建一个Woman类,并实现Person接口就行了。
通过反射机制,获得类的信息
在Java反射机制的定义中有:JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法。下面是一个实例:
import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; public class ClassUtil { /** * 获取成员函数的信息 */ public static void getClassMethodMessage(Object obj) { Class c = obj.getClass; //获取类的名称 //System.out.println(c.getName); /** * * Method类,方法对象 *getMethods方法获得所有public的方法,包括继承而来 *getDeclaredMethods是获取该类自己的声明的方法 */ Method ms = c.getMethods; for(int i = 0; i < ms.length; i++) { //得到方法的返回值类型的类类型 Class returnType = ms[i].getReturnType; System.out.print(returnType.getName + " "); //得到方法的名称 System.out.print(ms[i].getName + "("); //获取参数类型 Class paramTypes = ms[i].getParameterTypes; for(Class class1:paramTypes) { System.out.print(class1.getName + ","); } System.out.println(")"); } } public static void getFieldMessage(Object obj) { Class c = obj.getClass; /** * 成员变量也是对象 * java.lang.reflect.Field * */ Field fs = c.getDeclaredFields; for(Field field:fs) { //得到成员变量类型的类类型 Class fieldType = field.getType; String typeName = fieldType.getName; //得到成员变量的名称 String fieldName = field.getName; System.out.println(typeName + " " + fieldName); } } public static void printConMessage(Object obj) { Class c = obj.getClass; /** * 构造函数也是对象 * */ Constructor cs = c.getConstructors; for (Constructor constructor : cs) { System.out.print(constructor.getName + "("); //获取构造函数参数列表------>参数列表的参数类型 Class paramType = constructor.getParameterTypes; for (Class class1 : paramType) { System.out.print(class1.getName + ","); } System.out.println(")"); } } public static void main(String[] args) { String s = "hello"; getFieldMessage(s); ClassUtil.printConMessage(s); getClassMethodMessage(s); } }

方法的反射操作

可以通过方法的反射操作实现方法的调用:
import java.lang.reflect.Method; public class MethodDemo1 { public static void main(String[] args) { //要获取print(int, int) //要获取类的方法就要获取类的信息,获取类的信息就要获取类的类类型 A a1 = new A; Class c = a1.getClass; //2,获取方法 名称和参数列表 //getMethod获取的是public的方法 try { Method m = c.getDeclaredMethod("print", int.class,int.class); //方法的反射操作 //a1.print(10, 20);方法的反射操作,用m来进行方法调用和前者效果一致 Object obj = m.invoke(a1, 10,20);//如果方法有返回值返回值,没有就null } catch (Exception e) { e.printStackTrace; } } } class A { public void print(int a , int b) { System.out.println(a + b); } public void print(String a , String b) { System.out.println(a.toUpperCase + "," + b.toUpperCase); } }

以上是我今天对Java反射机制的学习总结,如果大家有更好的理解,欢迎在留言区评论交流,互相学习。

稿源:微信小程序开发www1.qixoo.com

JAVA反射机制—学习总结的更多相关文章

  1. Java反射机制——学习总结

    前几天上REST课,因为涉及到Java的反射机制,之前看过一直没有用过,有些遗忘了,周末找了些资料来重新学习,现在总结一下,加深印象. 什么是反射机制? 参考百度百科对java反射机制的定义: “JA ...

  2. Java反射机制学习与研究

    Java反射机制:可以获取正在运行时的Java对象. 1.判断运行时对象对象所属的类. 2.判断运行时对象所具有的成员变量和方法. 3.还可以调用到private方法,改变private变量的值. S ...

  3. Java反射机制学习

    Java 反射是Java语言的一个很重要的特征,它使得Java具体了“动态性”. 在Java运行时环境中,对于任意一个类,能否知道这个类有哪些属性和方法?对于任意一个对象,能否调用它的任意一个方法?答 ...

  4. JAVA反射机制学习随笔

    JAVA反射机制是用于在运行时动态的获取类的信息或者方法,属性,也可以用来动态的生成类,由于所有类都是CLASS的子类,我们可以用一个CLASS类的实例来实例化各种类 例如: Class<?&g ...

  5. java反射机制学习小结

    之前一直对java的反射机制理解得很模糊,今天因为学习spring,所以花了些时间总算把它理顺了,记录一下 另外,推荐读读这篇文章,写的挺好的http://blog.csdn.net/woshixuy ...

  6. java反射机制学习笔记

    内容引用自:https://www.cnblogs.com/wkrbky/p/6201098.html https://www.cnblogs.com/xumBlog/p/8882489.html,本 ...

  7. Java 反射机制学习资料

    Java反射——引言 Java反射——Class对象 Java反射——构造函数 Java反射——字段 Java反射——方法 Java反射——Getter和Setter Java反射——私有字段和私有方 ...

  8. java反射机制学习代码

    根据 http://www.iteye.com/topic/137944 文档进行学习 代码如下: package reflectTest; import java.lang.reflect.*; i ...

  9. Java反射机制的学习

    Java反射机制是Java语言被视为准动态语言的关键性质.Java反射机制的核心就是允许在运行时通过Java Reflection APIs来取得已知名字的class类的相关信息,动态地生成此类,并调 ...

随机推荐

  1. 运维工作中sed常规操作命令梳理

    sed是一个流编辑器(stream editor),一个非交互式的行编辑器.它一次处理一行内容.处理时,把当前处理的行存储在临时缓冲区中,称为"模式空间",接着用sed命令处理缓冲 ...

  2. js判断滚动条到底部

    判断滚动条到底部,需要用到DOM的三个属性值,即scrollTop.clientHeight.scrollHeight. scrollTop为滚动条在Y轴上的滚动距离. clientHeight为内容 ...

  3. 【转】【UML】使用Visual Studio 2010 Team System中的架构师工具(设计与建模)

    Lab 1: 应用程序建模 实验目标 这个实验的目的是展示如何在Visual Studio 2010旗舰版中进行应用程序建模.团队中的架构师会通过建模确定应用程序是否满足客户的需求. 你可以创建不同级 ...

  4. web页面的回流,认识与避免

    一.什么是回流? 回流是会导致页面重新渲染的一些元素,从而影响性能. 二.哪些因素会导致回流? 1.调整窗口的大小: 2.改变字体,如果用rem  设置了根目录的字体大小,这样就减少了回流的次数: 3 ...

  5. 【原创】Junit4详解一:Junit总体介绍

    Junit是一个可编写重复测试的简单框架,是基于Xunit架构的单元测试框架的实例.Junit4最大的改进是大量使用注解(元数据),很多实际执行过程都在Junit的后台做完了,而且写test case ...

  6. linux内核分析 第4章读书笔记

    第四章 进程调度 一.抢占与非抢占 1.非抢占式进程调度 进程会一直执行直到自己主动停止运行 2.抢占式进程调度 Linux/Unix使用的是抢占式的方式,强制的挂起进程的动作就叫做抢占. 二.进程优 ...

  7. 教你怎么fan qiang 进国内看视频而不受限制

    这一周难得今天闲一点,写日志也不知道写啥,谈爱情?...没得谈.思前想后还是在这儿带给大家点福利吧,主要是针对在国外生活的孩纸们看国内视频总是提示被限制的问题,qq音乐也是.     或许大家有听过f ...

  8. 采用指数退避算法实现ajax请求的重发,全部完成时触发回调函数

    目录: 0.Chrome扩展开发(Gmail附件管理助手)系列之〇——概述 1.Chrome扩展开发之一——Chrome扩展的文件结构 2.Chrome扩展开发之二——Chrome扩展中脚本的运行机制 ...

  9. C#出题库项目的总结(2)

    前记:好吧好吧,我好好的自我检讨,这个总结拖了这么久才来写,而且一周多没有看技术相关的东西,实在罪过,不过因为想做的事情太多,所以时间的分配确实是一个很严肃的问题,不是时间不够用,是我自己没有做好时间 ...

  10. 魅蓝Note2 在Android Studio 与 Eclipse中无法被检测到

    昨天到手的Note2 结果发现测试不了,一看魅蓝的版本是android 5.1,然后更新的自己的SDK. 最后…… 仍然不能识别到手机. ———————————— 今天在stackoverflow上搜 ...