Java中,反射是一种强大的工具。它使您能够创建灵活的代码,这些代码可以在运行时装配,无需在组件之间进行源代表链接。反射允许我们在编写与执行时,使我们的程序代码能够接入装载到JVM中的类的内部信息,而不是源代码中选定的类协作的代码。这使反射成为构建灵活的应用的主要工具。但需注意的是:如果使用不当,反射的成本很高。

  Java的类反射所需要的类并不多,它们分别是:Field、Constructor、Method、Class、Object,下面我将对这些类做一个简单的说明。
Field类:提供有关类或接口的属性的信息,以及对它的动态访问权限。反射的字段可能是一个类(静态)属性或实例属性,简单的理解可以把它看成一个封装反射类的属性的类。
Constructor类:提供关于类的单个构造方法的信息以及对它的访问权限。这个类和Field类不同,Field类封装了反射类的属性,而Constructor类则封装了反射类的构造方法。
Method类:提供关于类或接口上单独某个方法的信息。所反映的方法可能是类方法或实例方法(包括抽象方法)。 这个类不难理解,它是用来封装反射类方法的一个类。
Class类:类的实例表示正在运行的 Java 应用程序中的类和接口。枚举是一种类,注释是一种接口。每个数组属于被映射为 Class 对象的一个类,所有具有相同元素类型和维数的数组都共享该 Class 对象。
Object类:每个类都使用 Object 作为超类。所有对象(包括数组)都实现这个类的方法。

  下面举几个例子:

①、反射类中的属性

 1 package com.reflect;
2
3 import java.awt.event.ActionEvent;
4 import java.awt.event.ActionListener;
5 import java.lang.reflect.Field;
6
7 class A extends Object implements ActionListener{
8 public String s = “aaa”;
9 private int a = 3;
10 private Integer b = new Integer(4);
11 public A(){}
12 public A(int id,String name){}
13 private int abc(int id,String name){
14 return 0;
15 }
16 public void actionPerformed(ActionEvent e) {
17 System.out.println(“actionPerformed()”);
18 }
19 public void f1(int a){System.out.println(a);}
20 private void f2(int a){
21 System.out.println(a);
22 }
23 }
24
25 public class TestReflect {
26 public static void main(String[] args) throws SecurityException, NoSuchFieldException, IllegalArgumentException, IllegalAccessException {
27 A a = new A();
28 Class c = a.getClass();
29 //反射属性
30 Field[] fields = c.getFields();//得到公有属性
31 //Field[] fields = c.getDeclaredFields(); //得到所有属性
32 for (int i = 0; i < fields.length; i++) {
33 Class c1 = fields[i].getType(); //得到属性类型
34 System.out.println(c1);
35 }
36 System.out.println(“反射类中的私有属性”);
37 Field f = c.getDeclaredField(“a”);
38 f.setAccessible(true); //如果是私有成员,则要加这句,否则无法访问
39 System.out.println(f.get(a));//f.get(o)返回o对象的f属性
40 }
41
42 }

  反射类中的属性主要有两个方法:getFields()和getDeclaredFielda(),这两个方法返回的都是一个Field数组。getFields()得到的是public修饰的属性,而getDeclaredFielda()得到的是所有的属性。当然也可以得到某一个属性,如Field f = c.getDeclaredField(“a”),如果该属性是私有的话,必须加上f.setAccessible(true),否则用到的时候会报java.lang.IllegalAccessException异常。

②、反射类中的构造方法

 1 package com.reflect;
2
3 import java.lang.reflect.Constructor;
4
5 public class RefectConstructors {
6 public static void main(String[] args) throws NoSuchMethodException, SecurityException {
7 Class c = A.class;
8 String className = c.getName();//得到类名
9 Constructor[] cons = c.getConstructors(); //得到所有构造方法
10 for (int i = 0; i < cons.length; i++) {
11 Class[] parameterTypes = cons[i].getParameterTypes();
12 System.out.print(className+”(“);
13 for (int j = 0; j < parameterTypes.length; j++) {
14 System.out.print(parameterTypes[j].getName()+” “);
15 }
16 System.out.print(“)”);
17 System.out.println();
18 }
19 Constructor con = c.getConstructor(int.class,String.class); //得到某个具体的构造方法
20 System.out.println(con.getName());
21 }
22
23 }

  通过反射得到某个类的构造方法可以通过c.getConstructors()来获得,返回的是Constructor数组,也可以用c.getConstructor(Class …)方法来获得某个具体构造方法这个方法的参数是构造方法的参数类class对象。

③、反射接口和父类

 1 package com.reflect;
2
3 public class ReflectInterface {
4
5 public static void main(String[] args) {
6 Class c = A.class;
7 //反射得到所有的接口
8 Class[] interfaces = c.getInterfaces();
9 for (int i = 0; i < interfaces.length; i++) {
10 System.out.println(interfaces[i].getName());
11 }
12 //反射得到所有的父类
13 Class superClass = c.getSuperclass();
14 System.out.println(superClass.getName());
15 }
16 }

④、反射得到方法

 1 package com.reflect;
2
3 import java.lang.reflect.InvocationTargetException;
4 import java.lang.reflect.Method;
5
6 public class ReflectMethod {
7 public static void main(String[] args) throws SecurityException, NoSuchMethodException, IllegalArgumentException, IllegalAccessException, InvocationTargetException {
8 Class c = A.class;
9 A a = new A();
10 //Method[] methods = c.getDeclaredMethods(); //返回该类定义的所有方法。父类的方法若没有被重写则不会返回
11 Method[] methods = c.getMethods(); //得到public方法,包括从父类继承的方法
12 for (int i = 0; i < methods.length; i++) {
13 //输出方法的返回类型
14 System.out.print(methods[i].getReturnType());
15 //输出方法名
16 System.out.print(“ “+methods[i].getName()+”(“);
17 //获取方法的参数
18 Class[] paramTypes = methods[i].getParameterTypes();
19 for (int j = 0; j < paramTypes.length; j++) {
20 System.out.print(paramTypes[j].getName());
21 if(paramTypes.length>j+1){
22 System.out.print(“,”);
23 }
24 }
25 System.out.print(“)”);
26 System.out.println();
27 }
28 //得到指定私有方法名
29 Method m = c.getDeclaredMethod(“f2”,int.class); //方法名、方法中的参数
30 //私有方法需要让其能够被访问
31 m.setAccessible(true);
32 m.invoke(a, 5);
33 }
34
35 }

  c.getDeclaredMethods():返回该类定义的所有方法。父类的方法若没有被重写则不会返回。

  c.getMethods(): 得到所有public方法,包括从父类继承的方法

  c.getDeclaredMethod(String methodName,Class param) :得到某个具体的方法,要传入方法名和方法的参数。

转自http://www.liuling123.com/2013/05/java-reflex.html

java笔记十:java中的反射的更多相关文章

  1. 转 Java笔记:Java内存模型

    Java笔记:Java内存模型 2014.04.09 | Comments 1. 基本概念 <深入理解Java内存模型>详细讲解了java的内存模型,这里对其中的一些基本概念做个简单的笔记 ...

  2. Java基础学习笔记十 Java基础语法之final、static、匿名对象、内部类

    final关键字 继承的出现提高了代码的复用性,并方便开发.但随之也有问题,有些类在描述完之后,不想被继承,或者有些类中的部分方法功能是固定的,不想让子类重写.可是当子类继承了这些特殊类之后,就可以对 ...

  3. Java 笔记——在 IDEA 中使用 Maven 配置和使用 MyBatis

    1.前言 MyBatis 是什么? MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射. MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集. ...

  4. java 笔记(4) —— java I/O 流、字节流、字符流

    Java中使用流来处理程序的输入和输出操作,流是一个抽象的概念,封装了程序数据于输入输出设备交换的底层细节.JavaIO中又将流分为字节流和字符流,字节流主要用于处理诸如图像,音频视频等二进制格式数据 ...

  5. 《MFC游戏开发》笔记十 游戏中的碰撞检测进阶:地图类型&障碍物判定

    本系列文章由七十一雾央编写,转载请注明出处. http://blog.csdn.net/u011371356/article/details/9394465 作者:七十一雾央 新浪微博:http:// ...

  6. Java笔记:Java集合概述和Set集合

    本文主要是Java集合的概述和Set集合 1.Java集合概述 1)数组可以保存多个对象,但数组长度不可变,一旦在初始化数组时指定了数组长度,这个数组长度就是不可变的,如果需要保存数量变化的数据,数组 ...

  7. Java笔记:Java 流(Stream)、文件(File)和IO

    更新时间:2018-1-7 12:27:21 更多请查看在线文集:http://android.52fhy.com/java/index.html java.io 包几乎包含了所有操作输入.输出需要的 ...

  8. 菜鸡的Java笔记 简单JAVA 类的开发原则以及具体实现

    /*  现在要求定义一个雇员信息类 在这个类之中包含有雇员编号 姓名 职位 基本工资 佣金等信息    对于此时给定要求实际上就是描述一类事物,而这样的程序类在在java之中可以将其称为简单java类 ...

  9. Java笔记(十二)……类中各部分加载顺序及存放位置问题

    什么时候会加载类 使用到类中的内容时加载,三种情况: 创建对象:new StaticDemo(); 使用类中的静态成员:StaticCode.num = 9;  StaticCode.getNum() ...

随机推荐

  1. 经典算法——Jump Game(II)

    Given an array of non-negative integers, you are initially positioned at the first index of the arra ...

  2. [iOS]在NavigationController中的ScrollView中的子视图都会下移64个像素

    情况是这种: 我有一个UINavigationController,设置为self.window的root视图, 然后有一个UIVIewController是UINavigtionController ...

  3. hdu1698 Just a Hook (线段树区间更新 懒惰标记)

    Just a Hook Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Tota ...

  4. Java类集-set

    Set接口是Collection接口的子接口,Set接口中不能插入反复元素 Set接口的经常使用子类: HashSet是set接口的一个子类.特点:里面不能存放反复元素,并且採用散列的存储方式.所以没 ...

  5. lodash 检查值是否存在 includes

    _.includes(collection, value, [fromIndex=0]) 检查 值 是否在 集合中,如果集合是字符串,那么检查 值 是否在字符串中. 其他情况用 SameValueZe ...

  6. Android 4.4 Kitkat Phone工作流程浅析(七)__来电(MT)响铃流程

    本文来自http://blog.csdn.net/yihongyuelan 转载请务必注明出处 本文代码以MTK平台Android 4.4为分析对象,与Google原生AOSP有些许差异,请读者知悉. ...

  7. 冲销会计凭证:FBRP与FB08的区别

    FBRA一般是用来对已经清帐的凭证进行冲销的,我们知道普通的冲销只要用FB08就可以了,但是对于已经被清帐和凭证,我们只能用FBRA来进行清帐,我们来按以下的没测试来做验证一下.STEP1:用F-22 ...

  8. 关于SVN提交强制加入注释

    一.摘要 场景: 在这次开发项目过程中,团队中总是有人忘记添加注释. 问题: 1:其他成员不知道你提交是什么代码, 给回滚操作带来很多不必要的麻烦. 2:这个工作需要有一个人天天提醒大家在提交代码的时 ...

  9. 【SSH三大框架】Hibernate基础第九篇:cascade关联关系的级联操作

    这里要说的是Hibernate的关联关系的级联操作,使用cascade属性控制. 依旧用部门和员工举例.多个员工相应一个部门(多对一关联关系) 员工类:Employee.java package cn ...

  10. IDEA13中配置struts错误:ClassNotFoundException: org...dispatcher.ng.filter.StrutsPrepareAndExecuteFilter +找不到java程序包 解决办法

    问题一:ClassNotFoundException: org...dispatcher.ng.filter.StrutsPrepareAndExecuteFilter解决办法 1.确保所有strut ...