【软帝学院】Java反射机制!

Java反射

在 Java 中,并不是所有的类型信息都能在编译阶段明确,有一些类型信息需要在运行时才能确定,这种机制被称为 RTTI,即运行时类型识别,运行时类型识别主要由Class类实现。

1、 Class类

在Java中,我们常用“class”(首字母为小写的c)关键字来定义一个类,说这个类是对某一类对象的抽象。你比如说王二是一个网络知名作者,我们可以这样简单地定义作者类:

package com.cmower.java_demo.fifteen;

class Author {

private String pen_name;

private String real_name;}

现在,我们想知道Writer这个类本身的一些信息(比如说类名),该怎么办呢?这时候就需要用到“Class”(首字母为大写的C)类,该类包含了与类有关的信息。请看以下代码:

public class Test {

public static void main (String [] args) {

Author wanger = new Author();

Class c1 = wanger.getClass();

System.out.println(c1.getName());

//输出 com.cmower.java_demo.fifteen.Author

}}

当我们创建了作者对象wanger后,就可以通过wanger.getClass()获取wanger的Class对象,通过c1.getName()可获得wanger对象的类名。

想象一下,经过五年的刻意练习,张三从一名写作爱好者晋升为一名作家了。我们用代码来假装一下:

package com.cmower.java_demo.fifteen;

class Author {

private String pen_name;

private String real_name;}

class Writer extends Author {

private String honour;}

public class Test {

public static void main (String [] args) {

Author wanger = new Writer();

Class c1 = wanger.getClass();

System.out.println(c1.getName());

//输出 com.cmower.java_demo.fifteen.Writer

}}

在上例中,即使我们将Writer的对象引用wanger向上转型为Author,wanger的Class对象类型依然是Writer(通过输出结果可以判定)。这也就是说,Java能够在运行时自动识别类型的信息,它不会因为wanger的引用类型是Author而丢失wanger真正的类型信息(Writer)。Java是怎么做到这一点呢?

当Java创建某个类的对象,比如Writer类对象时,Java会检查内存中是否有相应的Class对象。如果内存中没有相应的Class对象,那么Java会在.class文件中寻找Writer类的定义,并加载Writer类的Class对象。

一旦Class对象加载成功,就可以用它来创建这种类型的所有对象。这也就是说,每个对象在运行时都会有对应的Class对象,这个Class对象包含了这个对象的类型信息。因此,我们能够通过Class对象知道某个对象“真正”的类型,并不会因为向上转型而丢失。

02、 获取Class对象的其他方式

在使用getClass()方法获取一个类的Class对象时,我们必须要先获取这个类的对象,比如上面提到的wanger。如果我们之前没有获取这个类的对象,就需要用另外两种方式来获取类的Class对象:

Class c2 = Writer.class;System.out.println(c2.getName());

try {

Class c3 = Class.forName("com.cmower.java_demo.fifteen.Writer");

System.out.println(c3.getName());} catch (ClassNotFoundException e) {

e.printStackTrace();}

1)当使用.class来获取Class对象时,不会自动地初始化该Class对象,初始化被延迟到了对静态方法或者非final静态域进行首次引用时才执行。这样做不仅更简单,而且更安全,因为它在编译时就会受到检查(因此不需要置于try语句块中)。

2)Class.forName会自动地初始化该Class对象,但需要指定类名,并且需要置于try语句块中。

03、 Class类提供的常用方法

Class类为我们提供了一些非常有用的方法,比如说getName()用来返回类名,getPackage()返回类所在的包名。

我们还可以利用Class类提供的newInstance()方法来创建相应类的对象,比如:

Class c2 = Writer.class;System.out.println(c2.getName());

try {

Writer wangsan = (Writer) c2.newInstance();

System.out.println(wangsan);

// 输出:[email protected]} catch (InstantiationException | IllegalAccessException e1) {

e1.printStackTrace();}

由于我们在创建Class对象c2时没有使用泛型,所以newInstance()返回的对象类型需要强转为Writer。我们可以在此基础上进行改进,示例如下:

Class<Writer> c4 = Writer.class;System.out.println(c4.getName());

try {

Writer wangsan = c4.newInstance();

System.out.println(wangsan);

// 输出:[email protected]} catch (InstantiationException | IllegalAccessException e1) {

e1.printStackTrace();}

04、 反射

我们还可以通过getFields()获取所有public修饰的字段,通过getMethods()返回所有public修饰的方法。

甚至,我们还可以通过getDeclaredFields()获取更多字段,包括公共、受保护、默认(包)访问和私有字段,但不包括继承字段。对应的,getDeclaredMethods()用来获取更多方法。示例如下:

package com.cmower.java_demo.fifteen;

import java.lang.reflect.Field;import java.lang.reflect.Method;

class Author {

private String pen_name;

private String real_name;}

class Writer extends Author {

private String honour;

private void makeMoney() {

System.out.println("很多很多钱");

}}

public class Test {

public static void main(String[] args) {

Class<Writer> c4 = Writer.class;

System.out.println(c4.getName());

try {

Writer wangsan = c4.newInstance();

System.out.println(wangsan);

Field[] fields = c4.getDeclaredFields();

for (Field field : fields) {

System.out.println(field.getName());

}

Method[] methods = c4.getDeclaredMethods();

for (Method method : methods) {

System.out.println(method.getName());

}

} catch (InstantiationException | IllegalAccessException e1) {

e1.printStackTrace();

}

}}

上面的例子其实涉及到了反射,Field、Method(还有例子中未提到的Constructor)都来自java.lang.reflect类库。Class类与java.lang.reflect类库一起对反射的概念进行了支持。

有时候,我们需要从磁盘文件或网络文件中读取一串字节码,并把它转换成一个类,这时候就需要用到反射。最常见的典型例子就是将一串JSON字符串(在网络传输中最初的形态可能是字节数组)反射为对应类型的对象。

阿里巴巴提供的FastJSON提供了 toJSONString() 和 parseObject() 方法来将 Java 对象与 JSON 相互转换。调用toJSONString方法即可将对象转换成 JSON 字符串,parseObject 方法则反过来将 JSON 字符串转换成对象。FastJSON的内部其实用的就是反射机制。

package com.cmower.common.util;

import java.io.UnsupportedEncodingException;

import org.apache.commons.logging.Log;import org.apache.commons.logging.LogFactory;

import com.alibaba.fastjson.JSON;

@SuppressWarnings("all")public class JsonUtil {

private static Log logger = LogFactory.getLog("json");

public static byte[] objectToByte(Object obj) throws UnsupportedEncodingException {

String jsonStr = JSON.toJSONString(obj);

logger.debug("序列化后数据:" + jsonStr);

return jsonStr.getBytes("UTF-8");

}

public static <T> T byteToObject(byte[] data, Class<T> obj) throws UnsupportedEncodingException {

String objectString = new String(data, "UTF-8");

logger.debug("反序列化后数据 : " + objectString);

return JSON.parseObject(objectString, obj);

}

public static <T> Object stringToObject(String data, Class<T> obj) throws UnsupportedEncodingException {

logger.debug("反序列化后数据 : " + data);

return JSON.parseObject(data, obj);

}}

原文地址:https://www.cnblogs.com/heqingxiaohuo/p/12172376.html

时间: 2024-11-13 09:24:45

【软帝学院】Java反射机制!的相关文章

软帝学院教你Java Applet基础

软帝学院教你Java Applet基础 applet是一种Java程序.它一般运行在支持Java的Web浏览器内.因为它有完整的Java API支持,所以applet是一个全功能的Java应用程序. 如下所示是独立的Java应用程序和applet程序之间重要的不同: Java中applet类继承了 java.applet.Applet类 Applet类没有定义main(),所以一个 Applet程序不会调用main()方法, Applets被设计为嵌入在一个HTML页面. 当用户浏览包含Appl

软帝学院教你用java编写计算器(二)

软帝学院教你用java编写计算器(二) import java.awt.BorderLayout; import java.awt.GridBagLayout; import java.awt.GridLayout; import java.awt.event.ActionEvent; import java.awt.event.ActionListener; import javax.swing.JButton; import javax.swing.JFrame; import javax.

Java反射机制知识点

反射机制 本文章参照:魔乐java视频教程讲解,感谢 1.认识反射 反射之中包含一个"反"的概念,所以就必须先从"正"开始解释,一般而言,当用户使用一个类的时候,应该先知道这个类,而后通过这个类产生实例化对象,但是"反"指的是通过对象找到类 package cn.test; class Person{};   // 定义一个Person类 public class ReflectDemo { public static void main(Str

软帝学院教你:初识springboot

软帝学院教你:初识springbootSpring Boot简化了基于Spring的应用开发.你可以很容易地创建一个独立的,产品 级别的Spring应用. 我们为Spring平台及第三方库提供开箱即用的设置,这样你就 可以有条不紊地开始.多数Spring Boot应用只需要很少的Spring配置. 接下来我们开始第一个springboot应用,首先需要安装Java SDK 1.8 或更高版本 第一步通过maven创建应用SpringBoot-First,编辑pom文件,在文件中加入spring-

java反射机制(一)—— 利用反射机制实例化对象

一.Java有着一个非常突出的动态相关机制:Reflection,用在Java身上指的是我们可以于运行时加载.探知.使用编译期间完全未知的classes.换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体.或对其fields设值.或唤起其methods.(度娘文库是这么说的) 二.这篇文章主要介绍一下通过反射机制去实例化一个类的对象,然后调用其方法.本文主要介绍两种方式,第一种就是通过构造函数来实例化,第二种就是通过Cl

【java】java反射机制,动态获取对象的属性和对应的参数值,并属性按照字典序排序,Field.setAccessible()方法的说明【可用于微信支付 签名生成】

方法1:通过get()方法获取属性值 package com.sxd.test.controller; public class FirstCa{ private Integer num; private String name; private Boolean flag; public Integer getNum() { return num; } public void setNum(Integer num) { this.num = num; } public String getNam

Java 反射机制

使用 Java 反射机制可以在运行时期检查 Java 类的信息,检查 Java 类的信息往往是你在使用 Java 反射机制的时候所做的第一件事情,通过获取类的信息你可以获取以下相关的内容: Class 对象 类名 修饰符 包信息 父类 实现的接口 构造器 方法 变量 注解 除了上述这些内容,还有很多的信息你可以通过反射机制获得,如果你想要知道全部的信息你可以查看相应的文档 JavaDoc for java.lang.Class 里面有详尽的描述. 在本节中我们会简短的涉及上述所提及的信息,上述的

Java反射机制浅析

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制. "程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言".从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C#不是动态语言.但是JAVA有着一个非常突出的动态相关机制:Reflection,用在Java身上指的是我们可以于运行时加载.探知.使用

java反射机制分析

本文转自:http://www.cnblogs.com/gulvzhe/archive/2012/01/27/2330001.html 浅显易懂,值得收藏 Java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象, 都能够调用它的任意一个方法和属性:这种动态获取的信息以及动态调用对象的方法的功能称为Java语言的反射机制.反射的概念是由Smith在1982年 首次提出的,主要是指程序可以访问.检测和修改它本身状态或行为的一种能力.这一概念的提出很快引发了

java 反射机制:运行时的类信息(为框架服务的Bug存在)

反射机制:JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法:这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制. 换一种引出反射的说法是:当通过反射与一个未知的类型的对象打交道是,JVM只是简单地检查这个类,看它是属于哪个特定的类(就想RTTI那样).在用它做其他事情之前必须先加载那个类的Class对象.因此,那个类的.class文件对于JVM来说必须是可获取的:那么在本地机器上,要么通过网络获得