Class 类

注意,这里的Class是大写的C。所以这里的Class是个特殊的类,而不是定义类的关键词class。

 

在Java中,每个class都有一个相应的Class对象。也就是说,当我们编写一个类,编译完成后,在生成的.class文件中,就会产生一个Class对象,用于表示这个类的类型信息。

 

构造Class实例的是三种方法:

1.       getClass()方法,必须生成目标类的是对象。

2.       forName(),会抛出异常,必须捕获。会运行目标类当中的静态代码段。

3.       .class属性来获取Class实例,应该是最简单的一种方式了。

对于基本数据类型的封装类,如果需要得到其封装的基本数据类型,还可以用.TYPE来获取相对应的基本数据类型的Class实例。

对于基本数据类型的封装类,如果不需要得到其封装的基本数据类型,而是需要获得封装类的Class类的实例,采用类名的.class的方式来获取Class实例。

 

publicclass ClassTest {
   publicstaticvoid main(String[] args){
 
      Point pt = new Point();
 
 
      Class c1 = pt.getClass();
      System.out.println("c1:"+ c1.getName());
   // c1:Point使用的是对象的.getClass()方法
 
      Class c2;
      try {
        c2 = Class.forName("Point");
        System.out.println("c2:"+ c2.getName());
// c2:Point使用的是Class类的静态方法,不需要生成类的实例。但是要注意的时候,这个方法会抛出一个ClassNotFoundException,需要用try catch捕获。
      } catch (ClassNotFoundException e) {
        //TODO Auto-generated catch block
        e.printStackTrace();
      }
      
      Class c3 = Point.class;
      System.out.println("c3:"+ c3.getName());
// c3:Point直接使用类名的.class属性获得Class对象。以上为Point类。在这种方法中,就不需要生成类的实例了。
 
      
      Class  c4 = int.class;
      System.out.println("c4:"+c4.getName());
// c4:int这个测试的实质和c3一致,只是类换成了int.
 
 
      Class c5 = Integer.TYPE;
      System.out.println("c5:"+ c5.getName());
//C5:int 这里需要注意,因为使用的是Integer的特殊用法,所以是用.TYPE来获取相对应的基本数据类型的Class实例。所以程序输出是int。如果需要得到Integer类的实例,参考c6
 
      Class c6 = Integer.class;
      System.out.println("c6:"+ c6.getName());
   }
}
//c6: java.lang.Integer 把integer当成一个普通的类来看待,不使用.TYPE属性,其实质和c3是一样的。
 
class Point{
   intx,y;
}
 
 
运行结果为:
c1:Point
c2:Point
c3:Point
c4:int
c5:int
c6:java.lang.Integer
 
在运行期间,如果我们要产生某个类的对象,Java虚拟机(JVM)会检查该类型的Class对象是否已被加载(ClassLoader)。如果没有被加载,JVM会根据类的名称找到.class文件并加载它。一旦某个类型的Class对象已被加载到内存,就可以用它来产生该类型的所有对象。
 
class ClassTest2
{
   publicstaticvoid main(String[] args)
   {  
      System.out.println("before new Point()");
      newPoint1();
      System.out.println("after new Point()");
      
      try
      {
        Class.forName("Line");
      }
      catch(Exception e)
      {
        e.printStackTrace();
      }
   
}
}
   
classPoint1
{
   static
   {
      System.out.println("Loading Point1");
   }
}
 
class Line
{
   static
   {
      System.out.println("Loading Line");
   }
}
 
运行结果为:
before new Point()
Loading Point1
after new Point()
Loading Line

 

考虑静态的代码段在什么时候执行,如果在main()方法当中,所有的类都被要加载,那么在new构造对象之前,Loading Point1和Loading Line都必须执行。结论是,在产生一个类的对象的时候,类才会被加载。或者说用Class.forName()去构造Class这个类的实例的时候,类才会被加载。

 

对Class类来说,可以使用newInstance()去动态创建目标类的实例。好处(用法):在不知道一个类的名字的时候,来创建一个类的实例。动态的去创建了一个类的实例,在创建的时候是不知道是哪个类的对象。

 

class ClassTest
{
   publicstaticvoid main(String[] args)
   {
        
      if(args.length!=1)
      {
        System.out.println("必须输入一个arg");
        return;
      }
      try
      {
        Class c=Class.forName(args[0]);  //通过命令行的参数,输入一个类(类名),那么c得到的是一个Class这个类的实例。
        Point pt = (Point)c.newInstance();通过c.newInstance()去创建一个Point类的实例(其实是object类型),并强制转换为Point类型。
        pt.output(); //然后就可以用pt作为普通的对象应用去使用了,和new出来的一样用。
      }
      catch(Exception e)
      {
        e.printStackTrace();
      }
   }
}
 
class Point
{
   static
   {
      System.out.println("Loading Point");
   }
   intx,y;
   void output()
   {
      System.out.println("x="+x+","+"y="+y);
   }
   /*Point(int x,int y)
   {
      this.x=x;
      this.y=y;
   }*/
}

要注意的是newInstance()只能使用目标类当中缺省的构造方法(不带任何参数的)。如果我们在目标类当中重写了各种构造方法,会抛出异常。

为了解决这个问题,为去了去使用带参数的构造方法,java提供了反射机制。