类
类是面向对象编程语言的一个重要概念,它是对一项事物的抽象概括,可以包含该事物的一些属性定义,以及操作属性的方法。面向对象编程中,我们都是以类来编码。
实例
简单理解,就是new,就是对类的实例化,创建这个类对应的实际对象,类只是对事物的描述,而实例化就相当于为这个描述新开辟了一块内存,可以改变这块区域里的各种属性(成员变量),当然,也可以实例化多块区域,只是不同的对象而已。
Class
注意这里C大写了,与类概念区分开,在java里,Class是一个实实在在的类,在包 java.lang 下,有这样一个Class.java文件,它跟我们自己定义的类一样,是一个实实在在的类,Class对象就是这个Class类的实例了。在Java里,所有的类的根源都是Object类,而Class也不例外,它是继承自Object的一个特殊的类,它内部可以记录类的成员、接口等信息,也就是在Java里,Class是一个用来表示类的类。(o(∩_∩)o 有点绕啊,抓住关键一点,Class是一个实实在在的类,可以为它创建实例,也就是本文后面提到的Class对象,也看叫做Class实例)。
java提供了下面几种获取到类的Class对象的方法:
1) 利用对象实例调用getClass()方法获取该对象的Class实例;
2) 使用Class类的静态方法forName("包名+类名"),用类的名字获取一个Class实例
3)运用 类名.class 的方式来获取Class实例;
我们知道java世界是运行在JVM之上的,我们编写的类代码,在经过编译器编译之后,会为每个类生成对应的.class文件,这个就是JVM可以加载执行的字节码。运行时期间,当我们需要实例化任何一个类时,JVM会首先尝试看看在内存中是否有这个类,如果有,那么会直接创建类实例;如果没有,那么就会根据类名去加载这个类,当加载一个类,或者当加载器(class loader)的defineClass()被JVM调用,便会为这个类产生一个Class对象(一个Class类的实例),用来表达这个类,该类的所有实例都共同拥有着这个Class对象,而且是唯一的。
总结
在java里,类只是信息描述的,写明了有哪些内部属性及接口,你可以理解为是定义了一套规则;而Class对象在java里被用来对类的情况进行表述的一个实例,也就是是类的实际表征,可以理解为是对规则的图表化,这样JVM才能直观的看懂,可以看做是一个模版;而类的实例化对象,就是通过模版,开辟出的一块内存进行实际的使用。
例子:
我们通过一个例子来理解Class实例,为了说明方便,我们新建一个包名深点的类。
新建Name.java(当然,该文件要放在com\dxjia\sample的目录下)
1 package com.dxjia.sample;
2
3 public class Name {
4 static int count = 0;
5 static {
6 count++;
7 System.out.println("Name Class Loaded! count = [" + count + "]" );
8 }
9
10 public Name() {
11 System.out.println("Name Constructor called!");
12 }
13
14 }
再在根目录新建一个Test主类
1 import com.dxjia.sample.Name;
2
3 public class Test {
4 static {
5 Name mName;
6 System.out.println("Test Class loaded");
7 }
8
9 public static void main(String[] args) {
10 System.out.println("entern Test main()");
11
12 // Name.class
13 Class mClassPointClass;
14 // Class.forName("完整包名+类名")
15 Class mClassForName;
16 // new 对象后,对象.getClass()
17 Class mClassObjectPointClass1;
18 Class mClassObjectPointClass2;
19
20 try {
21 //测试 类名.class
22 mClassPointClass = Name.class;
23 System.out.println("mClassPointClass = " + mClassPointClass);
24
25 //测试Class.forName()
26 mClassForName = Class.forName("com.dxjia.sample.Name");
27 System.out.println("mClassForName = " + mClassForName);
28
29 //测试Object.getClass()
30 Name name1 = new Name();
31 mClassObjectPointClass1 = name1.getClass();
32 System.out.println("mClassObjectPointClass1 = " + mClassObjectPointClass1);
33 } catch (ClassNotFoundException e) {
34 // TODO Auto-generated catch block
35 e.printStackTrace();
36 return;
37 }
38
39 Name name2;
40 System.out.println("defined one Name object");
41 name2 = new Name();
42 System.out.println("Name object instance done!");
43
44 mClassObjectPointClass2 = name2.getClass();
45
46 if (mClassForName == mClassPointClass
47 && mClassPointClass == mClassObjectPointClass1
48 && mClassObjectPointClass1 == mClassObjectPointClass2) {
49 System.out.println("all the Class object equal...");
50 }
51 }
52 }
分别对他们进行编译:
1 javac com\dxjia\sample\Name.java
2 javac Test.java
执行:
1 java Test
代码中使用了static静态代码块来进行实验,一个类的运行,JVM做会以下几件事情 1、类装载 2、链接 3、初始化 4、实例化;而初始化阶段做的事情是初始化静态变量和执行静态方法等的工作,而且永远只执行一次。
输出结果:
Test Class loaded
entern Test main()
mClassPointClass = class com.dxjia.sample.Name
Name Class Loaded! count = [1]
mClassForName = class com.dxjia.sample.Name
Name Constructor called!
mClassObjectPointClass1 = class com.dxjia.sample.Name
defined one Name object
Name Constructor called!
Name object instance done!
all the Class object equal...
通过结果可以看出在使用 类名.class获得Class实例时,并不会触发类的初始化,而 Class.forName方法就会触发,当然实例化对象肯定也是会触发的,但因为static代码块只执行一次,所以不会再有打印,最后的打印,说明一个类的Class实例只有唯一的一个。