[Java开发之路](14)反射机制

简介:
1. Class类

普通对象构造方式:
 
  
// 创建Book实例对象
Book book = new Book();
对于Class的实例对象如何构造呢?
Class的构造函数是私有的,只有JVM才能创建实例对象
 
  
// Class的构造函数是私有的,只有JVM才能创建Class实例对象
Class class1 = new Class(); // 错误
 
  
public final class Class<T> implements java.io.Serializable,
java.lang.reflect.GenericDeclaration,
java.lang.reflect.Type,
java.lang.reflect.AnnotatedElement {
 
/*
* Constructor. Only the Java Virtual Machine creates Class
* objects.
*/
private Class() {}
 
....
}

Class有三种表示方式:

(1)XXX.class  XXX为类名  
实际再告诉我们任何一个类都有一个隐含的已经太成员变量class
 
   
Class class1 = Book.class;
(2)XXX.getClass()  XXX为对象名称
已知该类的实例对象,通过getClass()方法获取
 
   
Book book = new Book();
Class class2 = book.getClass();
(3)通过Class类的forName方法获取  
 
   
try {
Class class3 = Class.forName("com.qunar.bean.Book");
System.out.println(class1 == class2);
System.out.println(class1 == class3);
} catch (ClassNotFoundException e) {
e.printStackTrace();
}

我们完全可以通过类的类类型创建该类的对象实例,通过class1, class2以及 class3创建Book的实例
 
  
try {
Class class3 = Class.forName("com.qunar.bean.Book");
// 通过类类型的newInstance方法创建实例对象
Book book2 = (Book)class3.newInstance();
book2.setPrice("23.4");
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}

2. Class动态加载类

Class.forName("类的全称")  不仅表示了类的类类型,还代表了动态加载类。
编译时刻加载类是静态加载类,运行时刻加载类是动态加载类。

3.反射

反射机制--用来检查可用的方法,并返回方法名。

人们想要在运行时获取类的信息的另一个动机,便是希望提供在跨网络的远程平台上创建和运行对象的能力。这被称为远程方法调用,它允许 一个Java程序将对象分布到多台机器上。

class类与java.lang.reflect类库一起对反射的概念进行了支持,该类库包含了Filed,Method以及Constructor类(每个类都实现了Member接口)。这些类型的对象是由JVM在运行时创建的,用以表示未知类里对应的成员。这样你就可以使用 Constructor创建新的对象,用get()和set()方法读取和修改与Field对象关联的字段,用invoke()方法调用与method对象关联的方法。另外,还可以调用getFields(),getMethods()和get Constructors()等很便利的方法,以返回表示字段,方法以及构造器的对象的数组。这样,匿名对象的类信息就能在运行时被完全确定下来,而在编译时不需要知道任何事情。

其实,反射机制并没有什么神奇之处。当通过反射与一个未知类型的对象打交道时,JVM只知道简单的检查这个对象,看它属于哪个特定的类。在用它做其他事情之前,我们必须先加载这个类的class对象。因此,那个类的.class文件对于JVM来说必须是可获取的,要么在本地机器上,要么可以通过网络可以获得。对于反射机制而言,在编译时不能取得.class文件,只能在运行时打开和检查.class文件。

3.1 获取方法信息

 
  
package com.qunar.reflect;
 
public class ReflectDemo {
 
public static void main(String[] args) {
// int的类类型
Class class1 = int.class;
Class class2 = String.class;
Class class3 = double.class;
Class class4 = Double.class;
Class class5 = void.class;
System.out.println("class1->" + class1.getName());
System.out.println("class2->" + class2.getName());
System.out.println("class3->" + class3.getName());
System.out.println("class4->" + class4.getName());
System.out.println("class5->" + class5.getName());
}
}
运行结果:

class1->int
class2->java.lang.String
class3->double
class4->java.lang.Double
class5->void

 
  
// 只打印简单名称(不包含包名称)
System.out.println("class2->" + class2.getSimpleName()); // String

 
  
/**
* 打印类成员方法信息(public函数,包括父类继承而来的)
* @param object
*/
public static void PrintClassPublicFunction(Object object){
// 获取类的信息,首先获取类的类类型
// 传递的是哪个子类的对象 c 就是该子类的类类型
Class c = object.getClass();
System.out.println("类的全称是:" + c.getName());
// 一个成员方法就是一个method对象
// getMethods方法是获取的是所有public的函数,包括父类继承而来的
Method[] methods = c.getMethods();
for (Method method : methods) {
// 获取方法返回值类型的类类型
Class returnType = method.getReturnType();
System.out.print(returnType.getName() + " ");
// 获取方法的名称
System.out.print(method.getName() + "(");
// 获取方法参数
// 得到方法参数列表中类型的类类型
Class[] paramTypes = method.getParameterTypes();
int size = paramTypes.length;
for (int i = 0;i < size;++i) {
if(i != 0){
System.out.print(",");
}//if
System.out.print(paramTypes[i].getName());
}//for
System.out.println(")");
}//for
}
运行结果:
boolean startsWith(java.lang.String)
boolean startsWith(java.lang.String,int)
java.lang.CharSequence subSequence(int,int)
java.lang.String substring(int,int)
java.lang.String substring(int)
[C toCharArray()
java.lang.String toLowerCase(java.util.Locale)
java.lang.String toLowerCase()
java.lang.String toUpperCase()
java.lang.String valueOf([C)
java.lang.Class getClass()
void notify()
void notifyAll()
void wait(long)
void wait(long,int)
void wait()
....

3.2 获取成员变量信息

 
  
/**
* 打印类成员变量信息
* @param object
*/
public static void PrintClassFiled(Object object){
// 获取类的信息,首先获取类的类类型
// 传递的是哪个子类的对象 c 就是该子类的类类型
Class c = object.getClass();
/* 成员变量也是对象,java.lang.reflect.Field 类封装了关于成员变量的操作
* getFields()方法获取的是所有的public的成员变量的信息
* getDeclaredFields()获取的是该类自己声明的成员变量的信息
*/
Field[] fields = c.getDeclaredFields();
for (Field field : fields) {
// 得到成员变量的类型的类类型
Class fieldType = field.getType();
// 得到成员变量的类型
System.out.print(fieldType.getName() + " ");
// 得到成员变量的名称
System.out.println(field.getName());
}//for
}
运行结果:

int MIN_VALUE
int MAX_VALUE
java.lang.Class TYPE
[C digits
[C DigitTens
[C DigitOnes
[I sizeTable
int value
int SIZE
long serialVersionUID
boolean $assertionsDisabled

3.3 获取构造函数信息

 
  
/**
* 打印类构造函数信息
* @param object
*/
public static void PrintClassConstructor(Object object){
// 获取类的信息,首先获取类的类类型
// 传递的是哪个子类的对象 c 就是该子类的类类型
Class c = object.getClass();
/* 构造函数也是对象,java.lang.reflect.Constructor 类封装了关于构造函数的操作
* getConstructors()方法获取的是所有的public的构造函数的信息
* getDeclaredConstructors()获取的是该类自己声明的构造函数的信息
*/
Constructor[] constructors = c.getConstructors();
for (Constructor constructor : constructors) {
// 构造函数的名称
System.out.print(constructor.getName() + "(");
// 获取构造函数的参数列表,得到的是参数列表的类类型
Class[] paramTypes = constructor.getParameterTypes();
int size = paramTypes.length;
for(int i = 0;i < size;++i){
if(i != 0){
System.out.print(",");
}//if
// 得到参数名称
System.out.print(paramTypes[i].getName());
}//for
System.out.println(")");
}//for
}
运行结果:

java.lang.String([B)
java.lang.String([B,int,int)
java.lang.String([B,java.nio.charset.Charset)
java.lang.String([B,java.lang.String)
java.lang.String([B,int,int,java.nio.charset.Charset)
java.lang.String(java.lang.StringBuilder)
java.lang.String(java.lang.StringBuffer)
java.lang.String([I,int,int)
java.lang.String([C,int,int)
java.lang.String([C)
java.lang.String(java.lang.String)
java.lang.String()
java.lang.String([B,int,int,java.lang.String)
java.lang.String([B,int)
java.lang.String([B,int,int,int)

3.4 方法的反射

如何获取某个方法?方法的名称和方法的参数列表才能唯一决定某个方法。
如何进行操作?通过method.invoke(对象,参数列表)

 
  
package com.qunar.reflect;
 
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
 
import com.qunar.bean.Calculates;
 
 
public class ReflectDemo {
 
public static void main(String[] args) {
// 获取类的信息,首先获取类的类类型
Calculates calculates = new Calculates();
calculates.setNum1(20);
calculates.setNum2(40);
Class c = calculates.getClass();
/* getMethod()方法获取的是public的方法信息
* getDeclaredMethod()获取的是该类自己声明的方法的信息
*/
try {
// 获取方法 名称和参数列表共同决定
// Method method = c.getDeclaredMethod("add", new Class[]{int.class,int.class});
Method method = c.getDeclaredMethod("add", int.class,int.class);
// 方法的反射
// 对于calculates.add(10,40)来说,方法的反射操作是用method方法调用 和 calculates.add(10,40)的效果一样
// int result = (int)method.invoke(calculates, new Object[]{10,40});
int result = (int)method.invoke(calculates, 10,40);
System.out.println(result);
// 对于没有参数的方法
Method method2 = c.getDeclaredMethod("print");
method2.invoke(calculates);
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
}






目录
相关文章
|
12天前
|
移动开发 前端开发 Java
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
JavaFX是Java的下一代图形用户界面工具包。JavaFX是一组图形和媒体API,我们可以用它们来创建和部署富客户端应用程序。 JavaFX允许开发人员快速构建丰富的跨平台应用程序,允许开发人员在单个编程接口中组合图形,动画和UI控件。本文详细介绍了JavaFx的常见用法,相信读完本教程你一定有所收获!
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
|
11天前
|
Java 数据库连接 Spring
反射-----浅解析(Java)
在java中,我们可以通过反射机制,知道任何一个类的成员变量(成员属性)和成员方法,也可以堆任何一个对象,调用这个对象的任何属性和方法,更进一步我们还可以修改部分信息和。
|
1月前
|
Java 开发者 微服务
Spring Boot 入门:简化 Java Web 开发的强大工具
Spring Boot 是一个开源的 Java 基础框架,用于创建独立、生产级别的基于Spring框架的应用程序。它旨在简化Spring应用的初始搭建以及开发过程。
59 6
Spring Boot 入门:简化 Java Web 开发的强大工具
|
23天前
|
存储 JavaScript 前端开发
基于 SpringBoot 和 Vue 开发校园点餐订餐外卖跑腿Java源码
一个非常实用的校园外卖系统,基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化,提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合,但并不是一个完全分离的项目。 前端视图通过JS的方式引入了Vue和Element UI,既能利用Vue的快速开发优势,
107 13
|
27天前
|
算法 Java API
如何使用Java开发获得淘宝商品描述API接口?
本文详细介绍如何使用Java开发调用淘宝商品描述API接口,涵盖从注册淘宝开放平台账号、阅读平台规则、创建应用并申请接口权限,到安装开发工具、配置开发环境、获取访问令牌,以及具体的Java代码实现和注意事项。通过遵循这些步骤,开发者可以高效地获取商品详情、描述及图片等信息,为项目和业务增添价值。
59 10
|
21天前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
57 2
|
30天前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
1月前
|
Java 程序员
深入理解Java异常处理机制
Java的异常处理是编程中的一块基石,它不仅保障了代码的健壮性,还提升了程序的可读性和可维护性。本文将深入浅出地探讨Java异常处理的核心概念、分类、处理策略以及最佳实践,旨在帮助读者建立正确的异常处理观念,提升编程效率和质量。
125 1
|
1月前
|
Java 开发者 UED
深入探索Java中的异常处理机制##
本文将带你深入了解Java语言中的异常处理机制,包括异常的分类、异常的捕获与处理、自定义异常的创建以及最佳实践。通过具体实例和代码演示,帮助你更好地理解和运用Java中的异常处理,提高程序的健壮性和可维护性。 ##
51 2
|
1月前
|
Java 开发者
Java中的异常处理机制深度剖析####
本文深入探讨了Java语言中异常处理的重要性、核心机制及其在实际编程中的应用策略,旨在帮助开发者更有效地编写健壮的代码。通过实例分析,揭示了try-catch-finally结构的最佳实践,以及如何利用自定义异常提升程序的可读性和维护性。此外,还简要介绍了Java 7引入的多异常捕获特性,为读者提供了一个全面而实用的异常处理指南。 ####
54 2