JAVA开发:深入研究java.lang.Class类
2016-01-29 09:34
197 查看
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)和关键字 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"),MyObject必须是接口或者类的名字。 调用forName("X") 将导致名为 X 的类被初始化。 3、获取Class类型对象的第三个方法非常简单。如果T是一个Java类型,那么T.class就代表了匹配的类对象。例如: Class cl1 = Manager.class; Class cl2 = int.class; Class cl3 = Double[].class; 注意:Class对象实际上描述的只是类型,而这类型未必是类或者接口。例如上面的int.class是一个Class类型的对象。由于历史原因,数组类型的getName方法会返回奇怪的名字。
我们总结如下: Class.forName("")返回的是类 Class.forName("").newInstance()返回的是object 有数据库开发经验朋友会发现,为什么在我们加载数据库驱动包的时候有的却没有调用newInstance( )方法呢?即有的jdbc连接数据库的写法里是Class.forName(xxx.xx.xx);而有一些:Class.forName(xxx.xx.xx).newInstance(),为什么会有这两种写法呢? 刚才提到,Class.forName("");的作用是要求JVM查找并加载指定的类,如果在类中有静态初始化器的话,JVM必然会执行该类的静态代码段。而在JDBC规范中明确要求这个Driver类必须向DriverManager注册自己,即任何一个JDBC Driver的Driver类的代码都必须类似如下: public class MyJDBCDriver implements Driver { static { DriverManager.registerDriver(new MyJDBCDriver()); } } 既然在静态初始化器的中已经进行了注册,所以我们在使用JDBC时只需要Class.forName(XXX.XXX);就可以了。 Java中工厂模式经常使用newInstance()方法来创建对象,因此从为什么要使用工厂模式上可以找到具体答案。 例如: class c = Class.forName("Example"); factory = (ExampleInterface)c.newInstance(); 其中ExampleInterface是Example的接口,可以写成如下形式: String className = "Example"; class c = Class.forName(className); factory = (ExampleInterface)c.newInstance(); 进一步可以写成如下形式: String className = readfromXMlConfig;//从xml 配置文件中获得字符串 class c = Class.forName(className); factory = (ExampleInterface)c.newInstance(); 上面代码已经不存在Example的类名称,它的优点是,无论Example类怎么变化,上述代码不变,甚至可以更换Example的兄弟类Example2 , Example3 , Example4……,只要他们继承ExampleInterface就可以。 从JVM的角度看,我们使用关键字new创建一个类的时候,这个类可以没有被加载。但是使用newInstance()方法的时候,就必须保证:1、这个 类已经加载;2、这个类已经连接了。而完成上面两个步骤的正是Class的静态方法forName()所完成的,这个静态方法调用了启动类加载器,即加载 java API的那个加载器。 现在可以看出,newInstance()实际上是把new这个方式分解为两步,即首先调用Class加载方法加载某个类,然后实例化。 这样分步的好处是显而易见的。我们可以在调用class的静态加载方法forName时获得更好的灵活性,提供给了一种降耦的手段。 最后用最简单的描述来区分new关键字和newInstance()方法的区别: newInstance: 弱类型。低效率。只能调用无参构造。 new: 强类型。相对高效。能调用任何public构造。
一、 getClass方法: 类型:public final Class<? extends Object> getClass() 功能:返回该对象的运行时类的java.lang.Class对象(API上的解释) 有方法类型可以知道,该方法只能由类的实例变量调用 例子: JButton b1 = new JButton("button1"); System.out.println(b1.getClass()); 输出: class javax.swing.JButton class属性 当你要获得一个类的Class对象时(作函数参数的时候),你不能调用getClass方法,那你只能用类名.class来达到效果 例子: System.out.println(JButton.class); 输出: class javax.swing.JButton getName方法: 类型:public String getName() 功能:以String形式返回次Class对象所表示的实体名称 例子: JButton b1 = new JButton("button1"); System.out.println(b1.getName()); 输出: javax.swing.JButton 可以发现用class属性和getClass返回的输出是一样的,用getName返回的比前面两种少了class和一个空格。 .eclipse工具 可以按"."然后马上提示很多方法 供你选择 那他如何知道"."了以后有哪些方法? 他用的语法就是getClass().getMethods();
相关文章推荐
- 1、Spring Boot简介
- 【Spring】java.lang.IndexOutOfBoundsException: Index: 256, Size: 256
- SpringMVC工作环境搭建 配置文件
- startexplorer插件——eclipse
- JMS--Spring整合JMS(一)——基于ActiveMQ实现
- eclipse中使用Lombok
- "XX cannot be resolved to a type "eclipse报错及解决方法参考
- java提高篇(十)-----详解匿名内部类 ,形参为什么要用final
- java字符串与二进制的相互转化
- 配置免安装版JAVA1.7的环境变量
- Java socket 学习
- Eclipse SVN插件冲突导致不能使用解决办法
- Eclipse插件—Easy Explorer
- groovy 线上修改Java程序的内存
- 浅析Java设计模式编程中的单例模式和简单工厂模式
- SpringMVC使用了@ResponseBody报406错误的问题(2)
- 【Struts2集锦】Struts2的Action中多个方法调用
- 王垠:给 Java 说句公道话
- SpringMVC,3种不同的URL路由配置方法(这根本不是一个小问题)
- myeclipse 常规web项目创建