在反射运用过程中,如果你想得到一个类的完整结构,那么就要使用到java.lang.reflect包中的几个类:

· Constructor  表示类中的构造方法

· Field  表示类中的属性

· Method 表示类中的方法

接下来,我们讲讲,如何通过这些API,去获取一个类的完整性结构。

首先,我们定义一个类:

  1. package com.chen.yuan.vo;
  2. interface China{ // 定义China接口
  3. public static final String NATIONAL = "China" ; // 定义全局常量
  4. public static final String AUTHOR = "李兴华" ; // 定义全局常量
  5. public void sayChina() ; // 无参的,没有返回值的方法
  6. public String sayHello(String name,int age) ; // 定义有两个参数的方法,并返回内容
  7. }
  8. public class Person implements China{
  9. private String name ;
  10. private int age ;
  11. public Person(){ // 无参构造
  12. }
  13. public Person(String name){
  14. this.name = name ; // 设置name属性
  15. }
  16. public Person(String name,int age){
  17. this(name) ;
  18. this.age = age ;
  19. }
  20. public void sayChina(){ // 覆写方法
  21. System.out.println("作者:" + AUTHOR + ",国籍:" + NATIONAL) ;
  22. }
  23. public String sayHello(String name,int age){
  24. return name + ",你好!我今年:" + age + "岁了!" ;
  25. }
  26. public void setName(String name){
  27. this.name = name ;
  28. }
  29. public void setAge(int age){
  30. this.age = age ;
  31. }
  32. public String getName(){
  33. return this.name ;
  34. }
  35. public int getAge(){
  36. return this.age ;
  37. }
  38. };

一、 通过反射取得类所实现的全部接口

  1. package com.chen.yuan.vo;
  2. public class GetInterfaceDemo{
  3. public static void main(String args[]){
  4. Class<?> c1 = null ; // 声明Class对象
  5. try{
  6. c1 = Class.forName("com.chen.yuan.vo.Person") ; // 实例化对象
  7. }catch(ClassNotFoundException e){
  8. e.printStackTrace() ;
  9. }
  10. Class<?> c[] = c1.getInterfaces() ; // 以数组形式返回实现的全部接口
  11. for(int i=0;i<c.length;i++){
  12. System.out.println("实现的接口名称:" + c[i].getName()) ; // 输出接口名称
  13. }
  14. }
  15. };

输出: 实现的接口名称:com.chen.yuan.vo.China

二、 取得类所继承的父类

  1. package com.chen.yuan.vo;
  2. public class GetSuperClassDemo{
  3. public static void main(String args[]){
  4. Class<?> c1 = null ; // 声明Class对象
  5. try{
  6. c1 = Class.forName("com.chen.yuan.vo.Person") ; // 实例化对象
  7. }catch(ClassNotFoundException e){
  8. e.printStackTrace() ;
  9. }
  10. Class<?> c2 = c1.getSuperclass() ; // 取得父类
  11. System.out.println("父类名称:" + c2.getName()) ;
  12. }
  13. };

输出: 父类名称:java.lang.Object

三、取得类中的全部构造方法

  1. public class GetConstructorDemo01{
  2. public static void main(String args[]){
  3. Class<?> c1 = null ; // 声明Class对象
  4. try{
  5. c1 = Class.forName("com.chen.yuan.vo.Person") ; // 实例化对象
  6. }catch(ClassNotFoundException e){
  7. e.printStackTrace() ;
  8. }
  9. Constructor<?> con[] = c1.getConstructors() ; // 取得一个类中的全部构造
  10. for(int i=0;i<con.length;i++){
  11. System.out.println("构造方法:" + con[i]) ; // 输出构造,直接打印
  12. }
  13. }
  14. };

输出:

构造方法:public com.chen.yuan.vo.Person(java.lang.String,int)

构造方法:public com.chen.yuan.vo.Person(java.lang.String)

构造方法:public com.chen.yuan.vo.Person()

以上直接打印出构造方法,调用的是Constructor类中 的toString()方法。

Constructor类中存在了一下几个几个方法:

· public int getModifiers()取得修饰符

· public String getName()取得方法名称

· public Class<?>[] getParameterTypes()取得参数的类型

  1. public class GetConstructorDemo02{
  2. public static void main(String args[]){
  3. Class<?> c1 = null ; // 声明Class对象
  4. try{
  5. c1 = Class.forName("com.chen.yuan.vo.Person") ; // 实例化对象
  6. }catch(ClassNotFoundException e){
  7. e.printStackTrace() ;
  8. }
  9. Constructor<?> con[] = c1.getConstructors() ; // 取得一个类中的全部构造
  10. for(int i=0;i<con.length;i++){
  11. Class<?> p[] = con[i].getParameterTypes() ; // 得到构造方法中的全部参数
  12. System.out.print("构造方法:" ) ; // 输出构造,直接打印
  13. System.out.print(con[i].getModifiers() + " ") ; // 得到修饰符
  14. System.out.print(con[i].getName()) ; // 取得构造方法的名字
  15. System.out.print("(") ;
  16. for(int j=0;j<p.length;j++){
  17. System.out.print(p[j].getName() + " arg" + i) ;
  18. if(j<p.length-1){
  19. // 判断此是否是最后一个参数
  20. System.out.print(","); // 输出“,”
  21. }
  22. }
  23. System.out.println("){}") ;
  24. }
  25. }
  26. };

输出:

构造方法:1 com.chen.yuan.vo.Person(java.lang.String arg0,int arg0){}

构造方法:1 com.chen.yuan.vo.Person(java.lang.String arg1){}

构造方法:1 com.chen.yuan.vo.Person(){}

观察输出,我们发现,所有的修饰符变成了数字,我们肯定不懂这些个数字是什么意思,那么接下来,让我们还原修饰符。此时,我们需要借助Modifier类完成,此类定义在java.lang.reflect包中,我们可以直接使用Modifier类中的toString(int mod)方法完成还原修饰符的操作。

先看一下Modifier类中的toString(int mod)方法的源码:

  1.  /**
  2.      * The {@code int} value representing the {@code public}
  3.      * modifier.
  4.      */
  5.     public static final int PUBLIC           = 0x00000001;
  6.  
  7.     /**
  8.      * The {@code int} value representing the {@code private}
  9.      * modifier.
  10.      */
  11.     public static final int PRIVATE          = 0x00000002;
  12.  
  13.     /**
  14.      * The {@code int} value representing the {@code protected}
  15.      * modifier.
  16.      */
  17.     public static final int PROTECTED        = 0x00000004;
  18.  
  19.     /**
  20.      * The {@code int} value representing the {@code static}
  21.      * modifier.
  22.      */
  23.     public static final int STATIC           = 0x00000008;
  24.  
  25.     /**
  26.      * The {@code int} value representing the {@code final}
  27.      * modifier.
  28.      */
  29.     public static final int FINAL            = 0x00000010;
  30.  
  31.     /**
  32.      * The {@code int} value representing the {@code synchronized}
  33.      * modifier.
  34.      */
  35.     public static final int SYNCHRONIZED     = 0x00000020;
  36.  
  37.     /**
  38.      * The {@code int} value representing the {@code volatile}
  39.      * modifier.
  40.      */
  41.     public static final int VOLATILE         = 0x00000040;
  42.  
  43.     /**
  44.      * The {@code int} value representing the {@code transient}
  45.      * modifier.
  46.      */
  47.     public static final int TRANSIENT        = 0x00000080;
  48.  
  49.     /**
  50.      * The {@code int} value representing the {@code native}
  51.      * modifier.
  52.      */
  53.     public static final int NATIVE           = 0x00000100;
  54.  
  55.     /**
  56.      * The {@code int} value representing the {@code interface}
  57.      * modifier.
  58.      */
  59.     public static final int INTERFACE        = 0x00000200; 
  1. public static String toString(int mod) {
  2. StringBuffer sb = new StringBuffer();
  3. int len;
  4.  
  5. if ((mod & PUBLIC) != 0) sb.append("public ");
  6. if ((mod & PROTECTED) != 0) sb.append("protected ");
  7. if ((mod & PRIVATE) != 0) sb.append("private ");
  8.  
  9. /* Canonical order */
  10. if ((mod & ABSTRACT) != 0) sb.append("abstract ");
  11. if ((mod & STATIC) != 0) sb.append("static ");
  12. if ((mod & FINAL) != 0) sb.append("final ");
  13. if ((mod & TRANSIENT) != 0) sb.append("transient ");
  14. if ((mod & VOLATILE) != 0) sb.append("volatile ");
  15. if ((mod & SYNCHRONIZED) != 0) sb.append("synchronized ");
  16. if ((mod & NATIVE) != 0) sb.append("native ");
  17. if ((mod & STRICT) != 0) sb.append("strictfp ");
  18. if ((mod & INTERFACE) != 0) sb.append("interface ");
  19.  
  20. if ((len = sb.length()) > 0) /* trim trailing space */
  21. return sb.toString().substring(0, len-1);
  22. return "";
  23. }

通过源码,我们应当很容易的发现,在Modifier类中的toString(int mod)方法中,我们进行了转换的操作。

  1. public class GetConstructorDemo03{
  2. public static void main(String args[]){
  3. Class<?> c1 = null ; // 声明Class对象
  4. try{
  5. c1 = Class.forName("com.chen.yuan.vo.Person") ; // 实例化对象
  6. }catch(ClassNotFoundException e){
  7. e.printStackTrace() ;
  8. }
  9. Constructor<?> con[] = c1.getConstructors() ; // 取得一个类中的全部构造
  10. for(int i=0;i<con.length;i++){
  11. Class<?> p[] = con[i].getParameterTypes() ; // 得到构造方法中的全部参数
  12. System.out.print("构造方法:" ) ; // 输出构造,直接打印
  13. int mo = con[i].getModifiers() ; // 得到所要的访问权限
  14. System.out.print(Modifier.toString(mo) + " ") ; // 得到修饰符
  15. System.out.print(con[i].getName()) ; // 取得构造方法的名字
  16. System.out.print("(") ;
  17. for(int j=0;j<p.length;j++){
  18. System.out.print(p[j].getName() + " arg" + i) ;
  19. if(j<p.length-1){
  20. // 判断此是否是最后一个参数
  21. System.out.print(","); // 输出“,”
  22. }
  23. }
  24. System.out.println("){}") ;
  25. }
  26. }
  27. };

输出:

构造方法:public com.chen.yuan.vo.Person(java.lang.String arg0,int arg0){}

构造方法:public com.chen.yuan.vo.Person(java.lang.String arg1){}

构造方法:public com.chen.yuan.vo.Person(){}

四、 取得全部方法

要想取得一个类中的全部方法,可以通过Class类中的getDeclaredMethods()方法,此方法返回一个Method类的对象数组,然后我们可以通过Method类,去进一步得到方法的详细信息。

  1. public class GetMethodDemo{
  2. public static void main(String args[]){
  3. Class<?> c1 = null ; // 声明Class对象
  4. try{
  5. c1 = Class.forName("com.chen.yuan.vo.Person") ; // 实例化对象
  6. }catch(ClassNotFoundException e){
  7. e.printStackTrace() ;
  8. }
  9. Method m[] = c1.getMethods() ; // 取得全部方法
  10. for(int i=0;i<m.length;i++){
  11. Class<?> r = m[i].getReturnType() ; // 得到返回值类型
  12. Class<?> p[] = m[i].getParameterTypes() ; // 取得全部参数的类型
  13. int xx = m[i].getModifiers() ; // 得到修饰符
  14. System.out.print(Modifier.toString(xx) + " ") ; // 输出修饰符
  15. System.out.print(r + " ") ;
  16. System.out.print(m[i].getName()) ;
  17. System.out.print("(") ;
  18. for(int j=0;j<p.length;j++){
  19. System.out.print(p[j].getName() + " " + "arg" + j) ;
  20. if(j<p.length-1){
  21. System.out.print(",") ;
  22. }
  23. }
  24. Class<?> ex[] = m[i].getExceptionTypes() ; // 取出异常
  25. if(ex.length>0){
  26. System.out.print(") throws ") ;
  27. }else{
  28. System.out.print(")") ;
  29. }
  30. for(int j=0;j<ex.length;j++){
  31. System.out.print(ex[j].getName()) ;
  32. if(j<p.length-1){
  33. System.out.print(",") ;
  34. }
  35. }
  36. System.out.println() ;
  37. }
  38. }
  39. };

输出;

public class java.lang.String getName()

public void setName(java.lang.String arg0)

public void sayChina()

public class java.lang.String sayHello(java.lang.String arg0,int arg1)

public void setAge(int arg0)

public int getAge()

public final void wait(long arg0,int arg1) throws java.lang.InterruptedException,

public final native void wait(long arg0) throws java.lang.InterruptedException

public final void wait() throws java.lang.InterruptedException

public boolean equals(java.lang.Object arg0)

public class java.lang.String toString()

public native int hashCode()

public final native class java.lang.Class getClass()

public final native void notify()

public final native void notifyAll()

五、 取得类中的属性

Class类中提供了两种不同的操作,可以获得Field:

1) 得到实现的接口或父类中的公共属性: public Field[] getFields() throws SecurityException

2) 得到本类中的全部属性: public Field[] getDeclaredFields() throws SecurityException

  1. public class GetFieldDemo{
  2. public static void main(String args[]){
  3. Class<?> c1 = null ; // 声明Class对象
  4. try{
  5. c1 = Class.forName("com.chen.yuan.vo.Person") ; // 实例化对象
  6. }catch(ClassNotFoundException e){
  7. e.printStackTrace() ;
  8. }
  9. { // 本类属性
  10. Field f[] = c1.getDeclaredFields() ; // 取得本类中的属性
  11. for(int i=0;i<f.length;i++){
  12. Class<?> r = f[i].getType() ; // 得到属性类型
  13. int mo = f[i].getModifiers() ; // 得到修饰符的数字
  14. String priv = Modifier.toString(mo) ; // 还原修饰符
  15. System.out.print("本类属性:") ;
  16. System.out.print(priv + " ") ;
  17. System.out.print(r.getName() + " ") ; // 得到属性类型
  18. System.out.print(f[i].getName()) ; // 输出属性名称
  19. System.out.println(" ;") ;
  20. }
  21. }
  22. { // 公共属性
  23. Field f[] = c1.getFields() ; // 取得本类中的公共属性
  24. for(int i=0;i<f.length;i++){
  25. Class<?> r = f[i].getType() ; // 得到属性类型
  26. int mo = f[i].getModifiers() ; // 得到修饰符的数字
  27. String priv = Modifier.toString(mo) ; // 还原修饰符
  28. System.out.print("公共属性:") ;
  29. System.out.print(priv + " ") ;
  30. System.out.print(r.getName() + " ") ; // 得到属性类型
  31. System.out.print(f[i].getName()) ; // 输出属性名称
  32. System.out.println(" ;") ;
  33. }
  34. }
  35. }
  36. };

输出:

本类属性:private java.lang.String name ;

本类属性:private int age ;

公共属性:public static final java.lang.String NATIONAL ;

公共属性:public static final java.lang.String AUTHOR ;

Java反射机制(二):通过反射取得类的结构的更多相关文章

  1. java 面向对象(二十):类的结构:代码块

    类的成员之四:代码块(初始化块)(重要性较属性.方法.构造器差一些)1.代码块的作用:用来初始化类.对象的信息2.分类:代码块要是使用修饰符,只能使用static分类:静态代码块 vs 非静态代码块3 ...

  2. java 中利用反射机制获取和设置实体类的属性值

    摘要: 在java编程中,我们经常不知道传入自己方法中的实体类中到底有哪些方法,或者,我们需要根据用户传入的不同的属性来给对象设置不同的属性值,那么,java自带的反射机制可以很方便的达到这种目的,同 ...

  3. 反射、反射机制、类加载、Class类专题复习

    一.反射概念 1.反射机制允许程序在执行期借助于ReflectionAPI取得任何类的内部信息(比如成员变量,构造器,成员方法等等),并能操作对象的属性及方法.反射在设计模式和框架底层都会用到. 2. ...

  4. 使用反射机制实现jQuery调用ashx类中的指定方法

    使用反射机制实现jQuery调用ashx类中的指定方法   近期用asp.net做个小网站,但又不喜欢使用asp.net的服务器端控件,经过一番思量后确定前端采用原始的html.后台采用Linq to ...

  5. JAVA基础(二)—— 常用的类与方法

    JAVA基础(二)-- 常用的类与方法 1 Math类 abs ceil floor 绝对值 大于等于该浮点数的最小整数 小于等于该浮点数的最大整数 max min round 两参数中较大的 两参数 ...

  6. 利用java反射机制 读取配置文件 实现动态类载入以及动态类型转换

    作者:54dabang 在spring的学习过程之中,我们能够看出通过配置文件来动态管理bean对象的优点(松耦合 能够让零散部分组成一个总体,而这些总体并不在意之间彼此的细节,从而达到了真正的物理上 ...

  7. 浅说Java中的反射机制(二)

    写过一篇Java中的反射机制,不算是写,应该是抄了,因为那是别人写的,这一篇也是别人写的,摘抄如下: 引自于Java基础--反射机制的知识点梳理,作者醉眼识朦胧.(()为我手记) 什么是反射? 正常编 ...

  8. JAVA中反射机制二

    声明:如需转载请说明地址来源:http://www.cnblogs.com/pony1223 反射二 利用反射创建对象 1.利用反射创建对象,首先我们创建一个类,类里面,我们知道构造函数有默认的构造函 ...

  9. java反射机制与动态加载类

    什么是java反射机制? 1.当程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言.我们认为java并不是动态语言,但是它却有一个非常突出的动态相关机制,俗称:反射. IT行业里这么说,没有 ...

  10. Java反射机制二 获取方法的返回值或参数的泛型信息

    在使用反射机制时,我们经常需要知道方法的参数和返回值类型,很简单  ,下面上示例,示例中的两个方法非常相似 package deadLockThread; import java.lang.refle ...

随机推荐

  1. JZOJ 平衡的子集

    Description 夏令营有N个人,每个人的力气为M(i).请大家从这N个人中选出若干人,如果这些人可以分成两组且两组力气之和完全相等,则称为一个合法的选法,问有多少种合法的选法? Input 第 ...

  2. HDU 3555 (递推&&记忆化)

    #include<stdio.h> #include<string.h> #define max 25 typedef __int64 LL; LL dp[max][]; // ...

  3. placeholder改变输入框字体颜色

    ::-webkit-input-placeholder {  color: #888;}:-moz-placeholder {  color: #888;}::-moz-placeholder{col ...

  4. SQL SERVER 2008 R2 插入数据非常慢

    表是5字段int类型,第一个字段是主健,自增字段 表结构: id int  Uncheckedbillno bigint  Uncheckedopid int  Checkedbillopid int ...

  5. nodeJs学习-16 数据字典示例

    1.banner ID title 标题 varchar(32) sub_title 副标题 varchar(16) src 图片地址 varchar(64) 2.文章 ID author 作者 va ...

  6. js树状菜单

    html部分 <ul class="tree"> <li><span><a href="#">JavaScrip ...

  7. 前端规范2-CSS规范

    CSS规范 缩进 使用Tab缩进(相当于四个空格) 选择器与{之间必须包含空格,参1 属性名和之后的:不允许包含空格,:与属性值之间必须包含空格.      例 列表性属性值在单行时,后必须跟一个空格 ...

  8. yum方式安装MySQL【转】

    在CentOS7中默认安装有MariaDB,这个是MySQL的分支,但为了需要,还是要在系统中安装MySQL,而且安装完成之后可以直接覆盖掉MariaDB. 另外至2919年5月4号, 默认安装的my ...

  9. Python学习之路14☞多线程与多进程

    一 进程与线程的概念 1.1 进程 进程定义: 进程就是一个程序在一个数据集上的一次动态执行过程.进程一般由程序.数据集.进程控制块三部分组成.我们编写的程序用来描述进程要完成哪些功能以及如何完成:数 ...

  10. hdu1536 sg打表

    标记数组用bool型防止超时.输入的f[ ]要排序. #include<stdio.h> #include<string.h> #include<algorithm> ...