什么是类反射

☆什么是反射

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为Java语言的反射机制。

反射(Reflection)是Java程序开发语言的特征之一,它允许运行中的Java程序对自身进行检查, 也称自审,并能直接操作程序的内部属性。例如,使用它能获得Java类中各成员的名称并显示出来。

Java的这一能力在实际应用中应用得很多,在其它的程序语言中根本就不存在这一特性。例如,Pascal、C或者C++中就没有办法在程序中获得函数定义相关的信息。

JavaBean是类反射的实际应用之一,它能让一些工具可视化的操作软件组件。这些工具通过类反射动态的载入并取得Java组件(类)的属性。后面学习的各种框架,基本上都会有反射的使用。

☆反射引例(HelloWorld、USB)

最简单的类反射:(相当于HelloWorld)

package cn.hncu.hello;import java.lang.reflect.Method;
import java.lang.reflect.Modifier;/*** 类反射的Helloworld版---入门* 只演示了类方法的类反射---Method[]* @author 陈浩翔* @version 1.0  2016-5-1*/
public class ReflectionHelloWold {public static void main(String[] args) {try {Class c = Class.forName("cn.hncu.introduce.Person");System.out.println(c);Method mc[] = c.getDeclaredMethods();//获得for(Method m:mc){//获得整个方法//包括修饰,返回类型,方法名字,方法参数System.out.println(m);//解剖Method对象System.out.println(Modifier.toString(m.getModifiers()));//修饰符System.out.println(m.getReturnType());//返回类型System.out.println(m.getName());//方法名字Class pt[] = m.getParameterTypes();for(Class p :pt){System.out.println("parameter = "+p);//参数}System.out.println();System.out.println(m.getDeclaringClass());//这个方法所在的类System.out.println();System.out.println("------------------------------");}} catch (ClassNotFoundException e) {e.printStackTrace();}}
}

Person类:

package cn.hncu.introduce;public class Person {private String name;int age;public final String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}private static final int sum(int age){return 0;}}

输出结果:

class cn.hncu.introduce.Person
private static final int cn.hncu.introduce.Person.sum(int)
private static final
int
sum
parameter = intclass cn.hncu.introduce.Person------------------------------
public int cn.hncu.introduce.Person.getAge()
public
int
getAgeclass cn.hncu.introduce.Person------------------------------
public void cn.hncu.introduce.Person.setAge(int)
public
void
setAge
parameter = intclass cn.hncu.introduce.Person------------------------------
public final java.lang.String cn.hncu.introduce.Person.getName()
public final
class java.lang.String
getNameclass cn.hncu.introduce.Person------------------------------
public void cn.hncu.introduce.Person.setName(java.lang.String)
public
void
setName
parameter = class java.lang.Stringclass cn.hncu.introduce.Person------------------------------

USB实例:反射最大的好处是解耦

USB接口:

package cn.hncu.usb.api;public interface USB {public abstract void work();
}

实现类:

package cn.hncu.usb.impl;import cn.hncu.usb.api.USB;public class UsbMouse implements USB{@Overridepublic void work() {System.out.println("UsbMouse....");}
}
package cn.hncu.usb.impl;import cn.hncu.usb.api.USB;public class UsbOne implements USB{@Overridepublic void work() {System.out.println("UsbOne...");}
}
package cn.hncu.usb.impl;import cn.hncu.usb.api.USB;public class UsbTwo implements USB{@Overridepublic void work() {System.out.println("UsbTwo....");}
}

工厂方法:!!!读取配置文件的在这里

package cn.hncu.usb.factory;import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.Properties;import cn.hncu.usb.api.USB;public class USBFactory {public static USB getUSB(){USB usb = null;Properties p = new Properties();//配置文件FileInputStream in;try {in = new FileInputStream("usb.config");//读取配置文件p.load(in);String className = p.getProperty("name");//通过name获得name后面=号后面的字符串,这样就可以通过修改配置文件来new不同的类Class c = Class.forName(className);usb = (USB) c.newInstance();//new 一个对象} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} catch (ClassNotFoundException e) {e.printStackTrace();} catch (InstantiationException e) {e.printStackTrace();} catch (IllegalAccessException e) {e.printStackTrace();}return usb;}
}

main方法:

package cn.hncu.usb;import cn.hncu.usb.api.USB;
import cn.hncu.usb.factory.USBFactory;public class Client {public static void main(String[] args) {USB usb = USBFactory.getUSB();usb.work();}}

配置文件:usb.config

你配置哪个,就是new哪个实现类的!!
‘#’号是配置文件的注释

#name = cn.hncu.usb.impl.UsbOne
#name = cn.hncu.usb.impl.UsbTwo
name = cn.hncu.usb.impl.UsbMouse

反射使用的三个步骤

用于反射的类,如Method,可以在java.lang.reflect包中找到。使用这些类的时候必须要遵循三个步骤:

第一步:获得你想操作的类的java.lang.Class对象。在运行中的Java程序中,用java.lang.Class类来描述类和接口等。

第二步:调用诸如getDeclaredMethods的方法,取得该类中定义的所有方法的列表。

第三步:使用反射的API来操作这些信息。

如下面这段代码:

Class c = Class.forName(“java.lang.String”);
Method ms[] = c.getDeclaredMethods();
System.out.println(ms[0].toString());

它将以文本方式打印出String中定义的第一个方法的原型。
☆反射示例(模拟instanceof的功能)

获取Class对象的三种方式

★ 方式一
通过对象的getClass方法进行获取。这种方式需要具体的类和该类的对象,以及调用getClass方法。
★ 方式二
任何数据类型(包括基本数据类型)都具备着一个静态的属性class,通过它可直接获取到该类型对应的Class对象。这种方式要使用具体的类,然后调用类中的静态属性class完成,无需调用方法,性能更好。
★ 方式三
通过Class.forName()方法获取。这种方式仅需使用类名,就可以获取该类的Class对象,更有利于扩展。

下面看怎么获取Class对象的代码:

package cn.hncu.reflect;/*** //获取Class对象的三种方式* @author 陈浩翔** @version 1.0  2016-5-1*/
public class ReflectGetClass {public static void main(String[] args) {//getClassObj1();//getClassObj2();//getClassObj3();test();}private static void test() {Person p = new Person();Object obj = (Object)p;Class c = obj.getClass();//运行时多态,new谁调谁System.out.println(c);//输出为:class cn.hncu.reflect.PersonSystem.out.println(int.class);//intSystem.out.println(Integer.class);//class java.lang.IntegerSystem.out.println(Integer.TYPE);//int System.out.println((int.class)==(Integer.class));//falseSystem.out.println((int.class)==(Integer.TYPE));//true}private static void getClassObj3() {try {String className = "cn.hncu.reflect.Person";Class c = Class.forName(className);//3333用字符串(类名)---依赖于String类System.out.println(c);//class cn.hncu.reflect.Person} catch (ClassNotFoundException e) {e.printStackTrace();}}private static void getClassObj2() {Class c = Person.class;//2222通过类名----依赖于Person类System.out.println(c);//class cn.hncu.reflect.Person}private static void getClassObj1() {Person p1 = new Person();Class c1 = p1.getClass();//1111通过对象System.out.println(c1);//class cn.hncu.reflect.PersonPerson p2 = new Person();Class c2 = p2.getClass();System.out.println(p1==p2);//false-不同的对象System.out.println(c1==c2);//true-同一个类模板,所有对象是共享同一个类模板的}}

运行结果在输出后面都注释了。。。

Class.isInstance(Object obj)判定指定的 Object 是否与此 Class 所表示的对象赋值兼容。

代码演示:

package cn.hncu.reflect;/*** @author 陈浩翔* @version 1.0  2016-5-1*/
public class SimulateInstanceof {/*** isInstance(Object obj)判定指定的 Object 是否与此 Class 所表示的对象赋值兼容。* @param args* @throws Exception*/public static void main(String[] args) throws Exception {Class c = Class.forName("cn.hncu.reflect.A");boolean boo = c.isInstance(new Integer(100));System.out.println(boo);//falseboolean boo2 = c.isInstance( new A() );System.out.println(boo2);//trueboolean boo3 = c.isInstance(new B());//B和A兼容//也就是说子类可以当父类用System.out.println(boo3);//true}
}
class A{private int age;
}class B extends A{private String name ;
}

类的解剖(获取类的定义信息)

Person类:

package cn.hncu.reflect;import java.io.IOException;public class Person extends Parent{String name;private int age;public int num;public static String schoolName;public Person(){}private Person(String name, int age){}protected Person(int age) throws IOException,NumberFormatException{}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}protected void setAge(int age) {this.age = age;}private int sum( int a){return 0;}private int sum(){return 0;}int aa(String a,int b) throws IOException,NumberFormatException{return 0;}
}class Parent{public static final int N=100;
}

★ 获取类的方法

找出一个类中定义了些什么方法,这是一个非常有价值也非常基础的反射用法。

代码演示:

package cn.hncu.reflect;import java.lang.reflect.Method;
import java.lang.reflect.Modifier;/*** 类的解剖(获取类的定义信息)* @author 陈浩翔** @version 1.0  2016-5-1*/
public class ReflectDecompose {private static final String CLASS_FILE_NAME="cn.hncu.reflect.Person";public static void main(String[] args) {try {fetchMethods(CLASS_FILE_NAME);//方法演示} catch (ClassNotFoundException e) {e.printStackTrace();}}/*** 方法演示* @param classFileName* @throws ClassNotFoundException*/private static void fetchMethods(String classFileName) throws ClassNotFoundException {Class c = Class.forName(CLASS_FILE_NAME);//1: 获取该类的能访问的所有public方法(包含当前类和父类的)Method methods[] = c.getMethods();//2: 获取Person类中声明的所有方法,包括私有和其它权限的//Method methods[] = c.getDeclaredMethods();for(int i=0;i<methods.length;i++){Method m = methods[i];//System.out.println("modifiers:"+ m.getModifiers());//int,这个是数字。方法的修饰符System.out.println("修饰符:"+ Modifier.toString( m.getModifiers() ) );//转换成字符串修饰符了。System.out.println("方法返回类型:"+ m.getReturnType());System.out.println("方法名称:"+ m.getName());Class paramTypes[] = m.getParameterTypes();//方法需要传入的参数类型-为空就是没有System.out.print("方法需要传入的参数类型: ");for(Class p:paramTypes){System.out.print(p+"   ");}System.out.println();Class exceptionTypes[] = m.getExceptionTypes();//方法抛出的异常System.out.print("方法抛出的异常: ");for(Class e:exceptionTypes){System.out.print(e+"   ");}System.out.println();System.out.println("此方法属于的类: "+m.getDeclaringClass());System.out.println();System.out.println("--------------------------------");}}}

运行结果太长了,自己可以运行看看!

★ 获取类的构造器

找出一个类中定义的构造方法,构造器没有返回类型。

代码演示:

package cn.hncu.reflect;import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;/*** 类的解剖(获取类的定义信息)* @author 陈浩翔** @version 1.0  2016-5-1*/
public class ReflectDecompose {private static final String CLASS_FILE_NAME="cn.hncu.reflect.Person";public static void main(String[] args) {try {//fetchMethods(CLASS_FILE_NAME);//方法演示fetchConstructors(CLASS_FILE_NAME);//类的构造方法演示} catch (ClassNotFoundException e) {e.printStackTrace();}}/*** 类的构造方法演示* @param classFileName* @throws ClassNotFoundException */private static void fetchConstructors(String classFileName) throws ClassNotFoundException {Class c = Class.forName(CLASS_FILE_NAME);//构造方法不能继承父类的!!!!!!!//1:  该类中所有public的构造方法。Constructor cons[] = c.getConstructors();//2: 获取该类中声明的所有构造方法,包括私有和其它权限的//Constructor cons[] = c.getDeclaredConstructors();for(int i=0;i<cons.length;i++){Constructor cs = cons[i];//System.out.println("modifiers:"+ m.getModifiers());//int,这个是数字。方法的修饰符System.out.println("修饰符:"+ Modifier.toString( cs.getModifiers() ) );//转换成字符串修饰符了。System.out.println("构造方法名称:"+ cs.getName());Class paramTypes[] = cs.getParameterTypes();//构造方法需要传入的参数类型-为空就是没有System.out.print("构造方法需要传入的参数类型: ");for(Class p:paramTypes){System.out.print(p+"   ");}System.out.println();Class exceptionTypes[] = cs.getExceptionTypes();//构造方法抛出的异常System.out.print("构造方法抛出的异常: ");for(Class e:exceptionTypes){System.out.print(e+"   ");}System.out.println();System.out.println("此构造方法属于的类: "+cs.getDeclaringClass());//全部是当前类!因为构造方法不能被继承System.out.println();System.out.println("--------------------------------");}}}

★ 获取类的属性字段

找出一个类中定义了哪些属性字段。

代码演示:

package cn.hncu.reflect;import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;/*** 类的解剖(获取类的定义信息)* @author 陈浩翔** @version 1.0  2016-5-1*/
public class ReflectDecompose {private static final String CLASS_FILE_NAME="cn.hncu.reflect.Person";public static void main(String[] args) {try {//fetchMethods(CLASS_FILE_NAME);//方法演示//fetchConstructors(CLASS_FILE_NAME);//类的构造方法演示fetchFields(CLASS_FILE_NAME);//类的成员变量演示} catch (ClassNotFoundException e) {e.printStackTrace();}}/*** 类的成员变量演示* @param classFileName* @throws ClassNotFoundException */private static void fetchFields(String classFileName) throws ClassNotFoundException {Class c = Class.forName(CLASS_FILE_NAME);//获取该类及其父类中的所有public成员变量(属性)Field fields[] = c.getFields();//获取该类声明的所有成员变量,包括私有及其它权限的//Field fields[] = c.getDeclaredFields();for(int i=0;i<fields.length;i++){Field f = fields[i];//System.out.println("modifiers:"+ m.getModifiers());//int,这个是数字。方法的修饰符System.out.println("修饰符:"+ Modifier.toString( f.getModifiers() ) );//转换成字符串修饰符了。System.out.println("方法名称:"+ f.getName());System.out.println("此构造方法属于的类: "+f.getDeclaringClass());//全部是当前类!因为构造方法不能被继承System.out.println();System.out.println("--------------------------------");}}
}

Class类真的很强大!

转载于:https://www.cnblogs.com/webmen/p/5739241.html

Java---类反射(1)---类反射入门和基础相关推荐

  1. 利用java反射调用类的的私有方法

    http://blog.csdn.net/sunyujia/article/details/2501709 今天和一位朋友谈到父类私有方法的调用问题,本来以为利用反射很轻松就可以实现,因为在反射看来根 ...

  2. 利用java反射调用类的的私有方法--转

    原文:http://blog.csdn.net/woshinia/article/details/11766567 1,今天和一位朋友谈到父类私有方法的调用问题,本来以为利用反射很轻松就可以实现,因为 ...

  3. Java 反射机制详解:私有方法调用头大?如何通过反射调用类中的私有方法?

    文章目录 前言 一.私有方法在本类中直接调用 1.1.在本类中实例化,调用私有方法 1.2.尝试在其他类直接调用私有方法(错误示范) 二.使用反射实例化类强制调用私有方法 2.1.使用类加载器加载被调 ...

  4. Java反射 Constructor类

    Java反射 Constructor类 Java通过反射可以获取构造方法,和使用构造方法创造对象. 在网上找了一个比较好理解的例子来弄清楚Constructor类. public Constructo ...

  5. Java 反射取类中类_Java反射机制(二):通过反射取得类的结构

    在反射运用过程中,如果你想得到一个类的完整结构,那么就要使用到java.lang.reflect包中的几个类: · Constructor  表示类中的构造方法 · Field  表示类中的属性 · ...

  6. Java知识点总结(反射-获取类的信息)

    Java知识点总结(反射-获取类的信息) @(Java知识点总结)[Java, 反射] 应用反射的API,获取类的信息(类的名字.属性.方法.构造器等) import java.lang.reflec ...

  7. java反射怎么获取结构体_java反射-使用反射获取类的所有信息

    在OOP(面向对象)语言中,最重要的一个概念就是:万事万物皆对象. 在java中,类也是一个对象,是java.lang.Class的实例对象,官网称该对象为类的类类型. Class 类的实例表示正在运 ...

  8. Java基础-通过反射获取类的信息

    目录 1.获取Field 2.反编译Field 3.通过反射机制访问对象属性 4.可变长度参数 5.反射Method(了解) 6.反编译(了解) 7.反射机制调用方法 8.反射Constructor ...

  9. java 反射 本类_Java 反射 Class类

    Java 反射 Class类 @author ixenos 摘要:Class类在反射中的地位.构造Class对象的三种方式.Class对象构造对应类型对象的三种方式 Class类在反射中的地位 位于j ...

  10. 反射获取类_新人也能看懂?如何使用 Java 反射?反射的用法及案例

    · 简介 Java Reflection,称为 Java 反射,是Java基础部分的一个比较难的点.Reflection(反射)是被视为动态语言的关键,通过反射机制,我们可以在运行时(runtime) ...

最新文章

  1. Domino URL Command 详解
  2. apache shiro jar包_只需要6个步骤,springboot集成shiro,并完成登录
  3. PHP 年龄计算函数
  4. Ubuntu常用翻译工具——星际译王StarDict
  5. 两道JVM面试题,竟让我回忆起了中学时代!
  6. 【零基础】讲述网络安全介绍
  7. NYOJ 116 士兵杀敌(二)
  8. jzoj3463-军训【双重嵌套二分,随机数据水法】
  9. 32位JDK和64位JDK
  10. 【.NET】XML文件的创建,修改,删除
  11. ともだちいじょう、こいびとみまん
  12. 能自动生成标注的APP,尝试一下吧
  13. 小马哥---高仿苹果6 plus 主板型号Q33 6582芯片刷机拆机主板图与开机界面图 镜面后壳
  14. wifi虚拟服务器怎么设置方法,虚拟路由器如何设置
  15. 天津大学计算机软件学院,2019计算机考研天津大学数据科学与服务工程团队(与软件学院共建)...
  16. 区块链在金融行业有哪些应用
  17. DHT11温湿度传感器——基于arduino
  18. opencv VideoWriter保存摄像头视频、本地视频等
  19. 网站服务器部署apk软件,供外网下载
  20. PS 2018CC安装完打开后,菜单界面的字体太大太小如何调整

热门文章

  1. spring教程笔记6
  2. struts教程笔记2
  3. 【图像处理】canny和sobel边缘检测__python-opencv
  4. “the import java.io cannot be resolved”错误的解决方法
  5. oracle java 映射_java程序访问映射后的oracle
  6. linux修改时区不用重启服务,Linux修改时区不用重启的方法
  7. 微信小程序云开发视频教程上线啦
  8. kubernetes [ERROR Swap]: running with swap on is not supported. Please disable swap
  9. 【mysql】Filesort on too many rows解决方法
  10. hashmap浅析-扩容