Java程序在执行时,Java执行时系统一直对全部的对象进行所谓的执行时类型标识。这项信息纪录了每一个对象所属的类。虚拟机通常使用执行时类型信息选准正确方法去执行,用来保存这些类型信息的类是Class类。Class类封装一个对象和接口执行时的状态,当装载类时。Class类型的对象自己主动创建。
Class 没有公共构造方法。Class 对象是在载入类时由Java 虚拟机以及通过调用类载入器中的 defineClass 方法自己主动构造的,因此不能显式地声明一个Class对象。
虚拟机为每种类型管理一个独一无二的Class对象。也就是说,每一个类(型)都有一个Class对象。执行程序时。Java虚拟机(JVM)首先检查是否所要载入的类相应的Class对象是否已经载入。假设没有载入,JVM就会依据类名查找.class文件,并将其Class对象载入。
主要的 Java 类型(boolean、byte、char、short、int、long、float 和 double)和keyword void 也都相应一个 Class 对象。
每一个数组属于被映射为 Class 对象的一个类,全部具有同样元素类型和维数的数组都共享该 Class 对象。
一般某个类的Class对象被加载内存,它就用来创建这个类的全部对象。
一、怎样得到Class的对象呢?有三种方法能够的获取:
1、调用Object类的getClass()方法来得到Class对象,这也是最常见的产生Class对象的方法。比如:
MyObject x;
Class c1 = x.getClass();
2、使用Class类的中静态forName()方法获得与字符串相应的Class对象。比如:
Class c2=Class.forName(“MyObject”),Employee必须是接口或者类的名字。
3、获取Class类型对象的第三个方法很easy。假设T是一个Java类型。那么T.class就代表了匹配的类对象。
比如
Class cl1 = Manager.class;
Class cl2 = int.class;
Class cl3 = Double[].class;
注意:Class对象实际上描写叙述的仅仅是类型。而这类型未必是类或者接口。
比如上面的int.class是一个Class类型的对象。
因为历史原因。数组类型的getName方法会返回奇怪的名字。
二、Class类的经常用法
一个Class对象描写叙述了一个特定类的属性,Class类中最经常使用的方法getName以 String 的形式返回此 Class 对象所表示的实体(类、接口、数组类、基本类型或 void)名称。
Class另一个实用的方法能够为类创建一个实例,这种方法叫做newInstance()。比如:
x.getClass.newInstance(),创建了一个同x一样类型的新实例。newInstance()方法调用默认构造器(无參数构造器)初始化新建对象。
返回该类的类载入器。
getComponentType()
返回表示数组组件类型的 Class。
getSuperclass()
返回表示此 Class 所表示的实体(类、接口、基本类型或 void)的超类的 Class。
isArray()
判定此 Class 对象是否表示一个数组类。
三、Class的一些使用技巧
forName和newInstance结合起来使用,能够依据存储在字符串中的类名创建对象。比如
Object obj = Class.forName(s).newInstance();
虚拟机为每种类型管理一个独一无二的Class对象。因此能够使用==操作符来比較类对象。比如:
if(e.getClass() == Employee.class)…
Class.forName():返回与给定的字符串名称相关联类或接口的Class对象。
Class.forName("");的作用是要求JVM查找并加载指定的类,假设在类中有静态初始化器的话,JVM必定会运行该类的静态代码段。
入参为:String className
getClass方法:
类型:public final Class extends Object> getClass()
功能:返回该对象的运行时类的java.lang.Class对象(API上的解释)
由方法类型可以知道,该方法只能由类的实例变量调用;
当你要获得一个类的Class对象时(作函数参数的时候),你不能调用getClass方法,那你只能用类名.class来达到效果
getName方法:
类型:public String getName()
功能:以String形式返回次Class对象所表示的实体名称
JButton b1 = new JButton("button1");
System.out.println(b1.getClass());
输出:class javax.swing.JButton
JButton b1 = new JButton("button1");
System.out.println(b1.getName());
输出:javax.swing.JButton
可以发现用class属性和getClass返回的输出是一样的,用getName返回的比前面两种少了class和一个空格。
//java中遍历实体类,获取属性名和属性值
public static void testReflect(Object model) throws Exception{
for (Field field : model.getClass().getDeclaredFields()) {
field.setAccessible(true);
System.out.println(field.getName() + ":" + field.get(model) );
}
}
通常都是使用java的关键字new来创建对象实例。
若有一个Something类,则可以通过下面的语句创建Something类的对象实例并指定到变量obj。
Something somethingNew = new Something();
通过new创建对象实例必须把类名写在源代码里面。
若程序写成如下,则可以根据当前对象(this)建立一个新实例对象(没有调用构造函数).
public class Something implements Cloneable{
private Something obj;
public Something cloneSomething()
{
try {
obj = (Something)this.clone();
// obj = (Something)clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return obj;
}
}
如果需要复制上面的那个obj指向的对象实例时,调用somethingNew.cloneSomething()方法就ok了。
但是为什么不直接使用somethingNew.clone()呢?
JDK中Object# clone()方法的原型是:
protected native Object clone() throws CloneNotSupportedException;
方法修饰符是protected,而不是public。这种访问的不可见性使得我们对Object#clone()方法不可见。
所以,必需重写Object的clone方法后才能使用。
public Object clone()throws CloneNotSupportedException
{
Something obj;
obj= (Something)super.clone();
return obj;
}
值得注意的是 :如果需要使用clone方法,必需实现java.lang.Cloneable接口,否则会抛出java.lang.CloneNotSupportedException。
另外clone方法所做的的操作是直接复制字段的内容,换句话说,这个操作并不管该字段对应的对象实例内容。
像这样字段对字段的拷贝(field to field copy)就成为"浅拷贝",clone方法所做的正是"浅拷贝".
利用java.lang.Class类的newInstance方法,则可根据Class对象的实例,建立该Class所表示的类的对象实例。
创建Something类的对象实例可以使用下面的语句(这样需要一个已经存在的对象实例)。
somethingNew.getClass().newInstance().
//或者使用下面的语句(只需要存在相应的.class文件即可)
Something instance = (Something) Class.forName("cn.softkid.test.Something").newInstance();
//如果包下不存在相应.class文件,则会抛出ClassNotFoundException。
注意 :newInstance创建对象实例的时候会调用无参的构造函数,所以必需确保类中有无参数的构造函数,否则将会
抛出java.lang.InstantiationException异常。无法进行实例化。
/**
* 解析制表的getDataSource方法,获取报表数据
* @param reportId
* @param branchNo
* @return
*/
@Override
public String exportTXTFile(String reportId, String branchNo, String coopBranchNo, String periodDate) {
Map<String, String> map = new HashMap<>();
// 根据报表xml获取templateType
String templateType = getTemplateTypeFromXml(reportId);
map.put(FieldValues.REPORT_ID, reportId);
map.put(FieldValues.BRANCH_NO, branchNo);
map.put(FieldValues.COOP_BRANCH_NO, coopBranchNo);
map.put(FieldValues.PERIOD_DATE, periodDate);
map.put(FieldValues.TEMPLATE_TYPE, templateType);
//通过制表平台获取报表内容
JSONObject jsonObject = uraReportbuilderService.getDataSource(map);
Map<String, Object> mapContent = jsonObject;
StringBuffer text = new StringBuffer();
for (Object keyValue : mapContent.values()) {
//keyValue中为ArrayList 代表该报表的DataSource(即数据库表)中的不同行;每一行都是一个实例化后的实体类
if (keyValue instanceof ArrayList) {
for (int i=0; i< ((ArrayList) keyValue).size(); i++) {
try{
Class ownClass = Class.forName(((ArrayList) keyValue).get(i).getClass().getName());
//这里需要用getClass().getName()
Field[] fields = ownClass.getDeclaredFields();
for (Field field : fields) {
field.setAccessible(true);
text.append(field.get(((ArrayList)keyValue).get(i)));
text.append("|");
}
//换行
text.append("\r\n");
} catch (Exception e){
logger.error("读取报表数据出错",e.toString());
return "fail";
}
}
}
}
return text.toString();
}