摘要:反射机制前言更多文章请一步本人博客网页版的的离线版什么是反射机制反射是语言的一个特性,它允程序在运行时注意不是编译的时候来进行自我检查并且对内部的成员进行操作。这个构造器就是用的反射在动态加载的时候来获取的中类的属性的。
Java反射机制 前言
什么是反射机制更多文章请一步本人博客https://chenjiabing666.github.io/
网页版的jdk的API
离线版API
主要的类反射是java语言的一个特性,它允程序在运行时(注意不是编译的时候)来进行自我检查并且对内部的成员进行操作。例如它允许一个java的类获取他所有的成员变量和方法并且显示出来。这个能特定我们不常看到,但是在其他的比如C或者C++语言中很不就存在这个特性。一个常见的例子是在JavaBean中,一些组件可以通过一个构造器来操作。这个构造器就是用的反射在动态加载的时候来获取的java中类的属性的。
ClassClass 类的实例表示正在运行的 Java 应用程序中的类和接口。Class没有公共的构造方法,Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的 defineClass 方法自动构造的
Constructor 提供关于类的单个构造方法的信息以及对它的访问权限(主要提供的是对构造方法使用)
Method 提供关于类或接口上多带带某个方法(以及如何访问该方法)的信息。所反映的方法可能是类方法或实例方法(包括抽象方法)
Field 主要提供对类中的成员变量的访问和使用
常用的方法Class类也使用了泛型,即是Class
这种形式的,可以直接使用一个具体的类传入,这样的话就不需要强制转换了,比如Class.newInstance()这样使用默认的构造方法构造一个对象就需要不再需要强制转换了即使用(ClassName)Class.newInstance()
获取Class的对象并且实例化getConstructor(Class[] params) 获取公共的(public)的构造方法,并且限定其中的参数个数和类型可以获得不同的公共构造方法
Constructor[] getConstructors() 返回所有的公共(public)的构造方法
getDeclaredConstructor(Class[] params) 获取所有指定的构造方法,但是需要注意的是当获取私有的构造方法的时候需要使用setAccessible设置访问权限为true才能进行构造,否则出现异常
Constructor[] getDeclaredConstructors() 返所有的构造方法包括public和private,protected修饰的
T newInstance() 返回的是一个调用默认的构造方法(public class_name())实例化的一个Object对象,如果使用泛型那么就返回T类型的,反之返回的是Object需要强制转换才能使用这个对象调用成员函数和成员变量
Class forName(String class_name) 返回class对象,每一个对都有一个方象法返回Class对象(test.class)
Package getPackage() 返回此类所在的包名(package demo) 当然也可以使用Package.getName()获得包的名字(demo)比如constructor.getPackage().getName()
int getModifiers() 返回的是类的修饰符的整数 类型(修饰符的类型有public private protected)其中得到整数可以使用Modifier中toString(int num)得到public,private,protected的类型,比如Modifier.toString(class1.getModifiers())
* Method getMethod(String name, Class>... parameterTypes) 返回指定参数的方法Method对象,注意这里仅仅是返回的时公共的方法(public) 比如:Method method=class1.getMethod("display",new Class[]{int.class})这里的display是方法的名字,有一个参数,类型为intMethod[] getMethods() 获取所有的公共的方法(public)返回的是一个数组(Method)
Method getDeclaredMethod(String name,Class>... parameterTypes)返回所有的指定的参数的方法(public,private,protected,但是不包括继承的),其中参数可以为null(无参数)
Method[] getDeclaredMethods() 获取所有的方法
Field getField(String name) 指定名字的公共成员变量(public)
Field[] getFields() 获取所有的公共的成员变量
Field getDeclaredField(String name) 获取所有的指定名称的成员变量(public,protected,private),同样在调用私有成员变量的时候需要先设置访问的权限,field.setAccessible(true)
Field[] getDeclaredFields() 获取所有的成员变量(public,protected,private)
getSuperclass() 返回表示此 Class 所表示的实体(类、接口、基本类型或 void)的超类的 Class。
URL getResource(String name) 查找指定名称的资源(图片,文件...)注意这个资源一定要和指定类在一个包中,否则返回null,比如查找Test类下的airplane.png图片:Test.class.getResource("airplane.png")这里返回的将是绝对路径
使用Class.forName(String className) 其中className一定是包含包的名字,下面的demo就是包的名字,Test是类的名字。这是最常用的方法,学过JDBC的都知道加载驱动的时候就是使用的Class.forName()
/* * 第一种使用forName(String className),其中className一定是包含包的名字,下面的demo就是包的名字,Test是类的名字 */ Class cls=Class.forName("demo.Test"); Test test=(Test)cls.newInstance(); //这里只是使用了默认的构造方法实例化对象
使用类名.class
Class cls=Test.class;
使用对象.getClass()
Test test=new Test(); Class cls=test.getClass();Constructor
常用的方法主要是用来对类的构造方法进行操作的,可以看出这个也使用了泛型,和上面的Class是一样的,注意这里如果没有使用泛型,那么原本放回T类型的现在都是返回Object
MethodT newInstance(Object parms) 使用带有参数的构造方法实例化对象,如果使用了泛型,那么返回的就是T类型的,反之返回的是Object类型的,需要强制转换
getName() 以字符串的形式返回构造方法的名称,具体的路径包含包名(demo.Test)
int getModifiers() 和Class类中的方法一样
常用的方法主要提供的是对类中的方法的操作
FieldObject invoke(Object obj,object args) 使用得到的Method对象调用方法,obj是类的已经构造好的对象,如果是静态方法直接写null,因为静态方法的调用不需要对象,返回值是Object类型的,如果接受返回值,需要使用强制转换成相应的类型,args是传入的参数,如果有多个参数,那么可以直接在后面用逗号添加或者直接创建数组new Object[]{22,"chenjiabing"}比如:method.invoke(test,22,"chenjiabing") method.invoke(test,new Object[]{22,"chenjiabing"})注意:如果调用的private类型的方法,那么需要在前面设置访问的权限,method.setAccessible(true)
String getName() 返回此方法的名字(display)
Modifier getModifiers() 返回此方法的修饰符的类型表示的整数(public,private...),可以使用Modifier.toString()转换成字符串形式
Class getReturnType() 返回这个方法的返回类型
String toString() 返回这个方法表示的字符串的形式
常用方法主要提供对类的成员变量的操作
ModifierString getName() 返回变量名字
Object get(Object obj) 返回此变量在指定对象中的值,因为在构造对象的时候每一个传入的变量的值都不一样,因此需要使用对象obj。obj表示传入的对象,返回的Object类型,因此需要强制转换
void set(Object obj,Object value) 改变obj对象上的变量的值为value
Modifier getModifiers() 返回整数表示修饰的类型
String getType() 获取变量的类型(int,String,double float.....)
常用的方法Modifier 类提供了 static 方法和常量,对类和成员访问修饰符进行解码。修饰符集被表示为整数,用不同的位位置 (bit position) 表示不同的修饰符。
实例static String toString(int mode) 将代表修饰符的整数形式转换为字符串形式的修饰符,比如将1转换成public
static isInterface(int mode) 如果整数参数包括 interface 修饰符,则返回 true,否则返回 false
static isStatic(int mode)
static isPrivate(int mode)
static isPublic(int mode)
static isAbstract(int mode)
Modifier.toString(Test.class.getModifiers()) //得到Test类的修饰符使用
有了上面的铺垫,我们就可以使用上面的这些类进行操作了,在进行操作之前,我们需要先定义一个类Test,放在demo包下,内容如下
package demo; import java.util.jar.Attributes.Name; import javax.print.attribute.standard.MediaSize.NA; public class Test { public String name; private int age; public Test() { this.name = "陈加兵"; this.age = 23; } public Test(String name, int age) { this.name = name; this.age = age; } public void display() { System.out.println("name=" + this.name + "----age=" + this.age); } public void set(String name, int age) { this.name = name; this.age = age; } private int getAge() { return this.age; } }实例化对象
使用Class默认的构造newInstance()
Class class1=Class.forName("demo.Test"); //静态加载Class Test test=(Test)class1.newInstance(); //调用默认的构造方法(public Test())实例化对象,由于没有使用泛型,因此需要强转 test.display(); //调用display方法
使用Class中的getConstructor()方法构造对象,需要注意的使用private类型构造方法时一定要先设置访问权限为true-constructor.setAccessible(true);
/* *调用public Test(String name,int age)得到Constructor的两种形式 * 1.Constructor constructor=class1.getConstructor(new Class[]{String.class,int.class}); * 2.Constructor constructor=class1.getConstructor(String.class,int.class);这个和上面的是一样的,就是使用的参数形式不一样 * * * * *使用newInstance()构造对象的两种方式 * 1.Test test=(Test)constructor.newInstance(new Object[]{"chenjiabing",22}); * 2.Test test=(Test)constructor.newInstance("chenjiabing",22); 只是形式不同而已,不过我还是喜欢上面的形式 * */ /* * 调用public Test(String name,int age) * Class.getConstructor()得到的是公共的构造方法,如果有私有的构造方法,那么就会报错,这时就要使用getDeclaredConstructor(Class>... parameterTypes) * Test test=(Test)constructor.newInstance("陈加兵",22); * * * 调用public Test() * Constructor constructor=class1.getConstructor(null); * Test test=(Test)constructor.newInstance(null); * * * 调用private Test(int age) * Constructor constructor=class1.getDeclaredConstructor(new Class[]{int.class}); constructor.setAccessible(true); //因为private类型是不可以直接访问的,因此需要设置访问权限为true Test test=(Test)constructor.newInstance(new Object[]{1000}); */ Class class1=Class.forName("demo.Test"); //访问public Test(String name,int age) // Constructor constructor=class1.getConstructor(new Class[]{String.class,int.class}); // Test test=(Test)constructor.newInstance("陈加兵",22); //访问默认的构造方法 // Constructor constructor=class1.getConstructor(null); // Test test=(Test)constructor.newInstance(null); //访问private类型的构造方法 Constructor constructor=class1.getDeclaredConstructor(new Class[]{int.class}); constructor.setAccessible(true); Test test=(Test)constructor.newInstance(new Object[]{1000}); test.display();成员方法的操作
使用Class.getMethod()和Class.getDeclaredMethod()方法获取方法,这两个方法的区别前面已经说过了,注意的是调用私有成员方法的之前一定要设置访问权限(method.setAccessible(true))
Method类中的其他方法前面也已经说过了,详细使用请自己尝试
/* * 获取Method对象的两种方式: * 1.Method method_set=class1.getMethod("set", new Class[]{String.class,int.class}); * 2.Method method_set=class1.getMethod("set", String.class,int.class); * * * 使用Method.invoke()调用方法的两种方式 * 1.Object o=method_set.invoke(test, new Object[]{"陈加兵",200}); * 2.Object object=method_set.invoke(test, "陈加兵",2000); */ /* * 获取公共方法(public): * 1.Method method=class1.getMethod("display",null); //public void display() * 2.Method method_set=class1.getMethod("set", new Class[]{String.class,int.class}); //获取public void set(String name,int age) * * * 获取私有方法(private,protected) * 1.Method method_getAge=class1.getDeclaredMethod("getAge", null); */ //使用构造方法构造一个Test对象 Class class1 =Class.forName("demo.Test"); Constructor成员变量的操作constructor=class1.getDeclaredConstructor(new Class[]{String.class,int.class}); Test test=constructor.newInstance(new Object[]{"陈加兵",22}); Method method=class1.getMethod("display",null); //获取public void display()方法的Method对象 Object obj=method.invoke(test, null); //调用方法display //获取public void set(String name,int age) // Method method_set=class1.getMethod("set", new Class[]{String.class,int.class}); Method method_set=class1.getMethod("set", String.class,int.class); // Object o=method_set.invoke(test, new Object[]{"陈加兵",200}); Object object=method_set.invoke(test, "陈加兵",2000); test.display(); //获取私有方法private int getAge() Method method_getAge=class1.getDeclaredMethod("getAge", null); method_getAge.setAccessible(true); //必须设置访问权限为true //判断返回值类型是否为int类型的 if("int".equals(method_getAge.getReturnType().toString())) { int ReturnData=(int) method_getAge.invoke(test, null); //调用并且获取返回值 System.out.println(ReturnData); }
主要使用的Field类,前面已经详细的说过了
/* * 获取public修饰的成员变量: * 1.Field field=class1.getField("name"); //获取public的成员变量name的Filed对象 * * 获取private,protected修饰的成员变量: * 1. Field field2=class1.getDeclaredField("age"); */ Class class1=Class.forName("demo.Test"); Test test=new Test("陈加兵",1000); Field field=class1.getField("name"); //获取public的成员变量name的Filed对象 System.out.println(field.get(test)); //获得test对象中的name属性的值 //获取private int age的Field对象 Field field2=class1.getDeclaredField("age"); field2.setAccessible(true); //设置访问权限 System.out.println(field2.get(test));参考文章
http://www.cnblogs.com/octobershiner/archive/2012/03/18/2404751.html
http://www.cnblogs.com/ixenos/p/5699420.html
https://yq.aliyun.com/wenzhang/show_17985
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/70113.html
近期在维护公司项目的时候遇到一个问题,因为实体类中的 set 方法涉及到了业务逻辑,因此在给对象赋值的过程中不能够使用 set 方法,为了实现功能,所以采用了反射的机制给对象属性赋值,借此机会也了解了反射的一些具体用法和使用场景,分以下两点对反射进行分析: 反射的优势和劣势 反射的应用场景 反射的优势和劣势 个人理解,反射机制实际上就是上帝模式,如果说方法的调用是 Java 正确的打开方式...
摘要:反射机制的应用实例在泛型为的中存放一个类型的对象。工厂模式可以参考现在我们利用反射机制实现工厂模式,可以在不修改工厂类的情况下添加任意多个子类。 学习交流群:669823128java 反射 定义 功能 示例概要:Java反射机制详解| |目录 1反射机制是什么 2反射机制能做什么 3反射机制的相关API 通过一个对象获得完整的包名和类名 实例化Class类对象 获取一个对象的父类与...
摘要:反射机制是什么反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法对于任意一个对象,都能够调用它的任意一个方法和属性这种动态获取的信息以及动态调用对象的方法的功能称为语言的反射机制反射机制能做什么反射机制主要提供了以下功 反射机制是什么 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种...
摘要:反射机制一结合官方通过编写的反射教程,复习一下反射的知识。反射的概念反射是一种在运行时获取以及修改应用行为的一种工具。因为反射需要动态的解析类的信息,相比于非反射使用的方式要慢。反射需要获取一定的运行时权限,在特定的安全环境下不一定存在。 Java反射机制(一) 结合Oracle官方通过JDK8编写的反射教程,复习一下反射的知识。结尾篇补一个小例子。 主要内容 这次博客的主要内容就是简...
摘要:与都继承自类,在中也是使用字符数组保存字符串,,这两种对象都是可变的。采用字节码的好处语言通过字节码的方式,在一定程度上解决了传统解释型语言执行效率低的问题,同时又保留了解释型语言可移植的特点。 String和StringBuffer、StringBuilder的区别是什么?String为什么是不可变的? String和StringBuffer、StringBuilder的区别 可变性...
阅读 827·2023-04-25 23:59
阅读 3621·2021-10-08 10:04
阅读 1655·2019-08-30 14:05
阅读 985·2019-08-30 13:58
阅读 459·2019-08-29 18:41
阅读 1075·2019-08-29 17:15
阅读 2303·2019-08-29 14:13
阅读 2727·2019-08-29 13:27