Java 如何在运行时识别类型信息?,java 发展史百度百科
一旦 Class 对象加载成功,就可以用它来创建这种类型的所有对象。这也就是说,每个对象在运行时都会有对应的 Class 对象,这个 Class 对象包含了这个对象的类型信息。因此,我们能够通过 Class 对象知道某个对象“真正”的类型,并不会因为向上转型而丢失。
二、获取 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 语句块中。
三、Class 类提供的常用方法
===============
Class 类为我们提供了一些非常有用的方法,比如说 getName()用来返回类名,getPackage()返回类所在的包名。
我们还可以利用 Class 类提供的 n
ewInstance()方法来创建相应类的对象,比如:
Class c2 = Writer.class;
System.out.println(c2.getName());
try {
Writer wangsan = (Writer) c2.newInstance();
System.out.println(wangsan);
// 输出:com.cmower.java_demo.fifteen.Writer@7852e922
} 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);
// 输出:com.cmower.java_demo.fifteen.Writer@7852e922
} catch (InstantiationException | IllegalAccessException e1) {
e1.printStackTrace();
}
四、反射
====
我们还可以通过 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);
}
}
评论