class对象

来源:互联网 发布:人工智能弹钢琴 编辑:程序博客网 时间:2024/05/02 01:34
 java中把生成Class对象和实例对象弄混了,更何况生成Class对象和生成instance都有多种方式。所以只有弄清其中的原理,才可以深入理解。首先要生成Class对象,然后再生成Instance。那Class对象的生成方式有哪些呢,以及其中是如何秘密生成的呢?

Class对象的生成方式如下:

1.Class.forName("类名字符串")  (注意:类名字符串必须是全称,包名+类名)

2.类名.class

3.实例对象.getClass()

通过一段小程序,来观察一下Class对象的生成的原理。

[java] view plaincopyprint?
  1. /** 
  2.  
  3.  * 2012-2-6 
  4.  
  5.  * Administrator 
  6.  
  7.  */  
  8.   
  9. /** 
  10.  
  11.  * @author: 梁焕月  
  12.  
  13.  * 文件名:TestClass.java  
  14.  
  15.  * 时间:2012-2-6上午10:01:52   
  16.  
  17.  */  
  18.   
  19. public class TestClass {  
  20.   
  21.    
  22.   
  23. public  static void main(String[] args)  
  24.   
  25. {  
  26.   
  27. try {  
  28.   
  29. //测试Class.forName()  
  30.   
  31. Class testTypeForName=Class.forName("TestClassType");          
  32.   
  33. System.out.println("testForName---"+testTypeForName);  
  34.   
  35. //测试类名.class  
  36.   
  37. Class testTypeClass=TestClassType.class;  
  38.   
  39. System.out.println("testTypeClass---"+testTypeClass);  
  40.   
  41. //测试Object.getClass()  
  42.   
  43. TestClassType testGetClass= new TestClassType();  
  44.   
  45. System.out.println("testGetClass---"+testGetClass.getClass());  
  46.   
  47.    
  48.   
  49. catch (ClassNotFoundException e) {  
  50.   
  51. // TODO Auto-generated catch block  
  52.   
  53. e.printStackTrace();  
  54.   
  55. }  
  56.   
  57.    
  58.   
  59. }  
  60.   
  61. }  
  62.   
  63.  class TestClassType{  
  64.   
  65. //构造函数  
  66.   
  67. public TestClassType(){  
  68.   
  69. System.out.println("----构造函数---");  
  70.   
  71. }  
  72.   
  73. //静态的参数初始化  
  74.   
  75. static{  
  76.   
  77. System.out.println("---静态的参数初始化---");  
  78.   
  79. }  
  80.   
  81. //非静态的参数初始化  
  82.   
  83. {  
  84.   
  85. System.out.println("----非静态的参数初始化---");  
  86.   
  87. }          
  88.   
  89. }  

 

测试的结果如下:

---静态的参数初始化---

testForName---class TestClassType

testTypeClass---class TestClassType

----非静态的参数初始化---

----构造函数---

testGetClass---class TestClassType

 

根据结果可以发现,三种生成的Class对象一样的。并且三种生成Class对象只打印一次“静态的参数初始化”。 

我们知道,静态的方法属性初始化,是在加载类的时候初始化。而非静态方法属性初始化,是new类实例对象的时候加载。

因此,这段程序说明,三种方式生成Class对象,其实只有一个Class对象。在生成Class对象的时候,首先判断内存中是否已经加载。

所以,生成Class对象的过程其实是如此的:

当我们编写一个新的java类时,JVM就会帮我们编译成class对象,存放在同名的.class文件中。在运行时,当需要生成这个类的对象,JVM就会检查此类是否已经装载内存中。若是没有装载,则把.class文件装入到内存中。若是装载,则根据class文件生成实例对象。



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构造。

 

.class与getClass()

.class其实是在java运行时就加载进去的

getClass()是运行程序时动态加载的

Java代码  收藏代码
  1. public class ExtendClass extends Baseclass {  
  2.     private String width;  
  3.     public String getWidth() {  
  4.         return width;  
  5.     }  
  6.     public void setWidth(String width) {  
  7.         this.width = width;  
  8.     }  
  9.     public static void main(String[] arg0) {  
  10.         Baseclass baseclass1 = new ExtendClass();  
  11.         Baseclass baseclass2 = new Baseclass();  
  12.         System.out.println(baseclass1.getClass().getSimpleName());// 实际运行的是继承类Extendclass  
  13.         System.out.println(baseclass2.getClass().getSimpleName());// 实际运行的是Baseclass  
  14.         System.out.println(Baseclass.class.getSimpleName());// 加载时类名  
  15.         System.out.println(ExtendClass.class.getSimpleName());// 加载时类名  
  16.     }  
  17. }  
  18. class Baseclass {  
  19.     private String height;  
  20.     public String getHeight() {  
  21.         return height;  
  22.     }  
  23.     public void setHeight(String height) {  
  24.         this.height = height;  
  25.     }  
  26. }  

 执行结果为:

ExtendClass
Baseclass
Baseclass
ExtendClass


0 0