Java中的Class類(lei)
時間:2018-09-27 來源:未知
Java的(de)(de)類(lei)庫日益龐大,所(suo)包含的(de)(de)類(lei)和(he)接口也不(bu)計其(qi)數。但其(qi)中(zhong)有一些(xie)非常(chang)重要的(de)(de)類(lei)和(he)接口,是(shi)(shi)Java類(lei)庫中(zhong)的(de)(de)核心部分。常(chang)見的(de)(de)有String、Object、Class、Collection、ClassLoader...,熟悉這(zhe)些(xie)類(lei)是(shi)(shi)學好Java的(de)(de)基(ji)礎。而這(zhe)些(xie)類(lei)一般不(bu)容易理解,需要做深入的(de)(de)研(yan)究和(he)實踐才能掌握。下面是(shi)(shi)我對這(zhe)些(xie)類(lei)理解和(he)使用(yong)的(de)(de)一些(xie)總結。歡(huan)迎你在(zai)閱(yue)讀后將(jiang)你寶貴的(de)(de)意見和(he)讀后感(gan)留下!
本文引用地址://fsbing.cn/emb/Column/7448.html
Java程(cheng)序在運(yun)(yun)行時,Java運(yun)(yun)行時系(xi)統一直對(dui)(dui)所有的(de)對(dui)(dui)象(xiang)進行所謂的(de)運(yun)(yun)行時類(lei)(lei)型(xing)標識。這項信(xin)息紀錄了每個(ge)對(dui)(dui)象(xiang)所屬的(de)類(lei)(lei)。虛擬機通常(chang)使(shi)用運(yun)(yun)行時類(lei)(lei)型(xing)信(xin)息選(xuan)準正確(que)方法去執行,用來保存這些(xie)類(lei)(lei)型(xing)信(xin)息的(de)類(lei)(lei)是Class類(lei)(lei)。Class類(lei)(lei)封裝(zhuang)一個(ge)對(dui)(dui)象(xiang)和接口運(yun)(yun)行時的(de)狀態(tai),當(dang)裝(zhuang)載類(lei)(lei)時,Class類(lei)(lei)型(xing)的(de)對(dui)(dui)象(xiang)自動創建。
Class 沒有公共構(gou)造(zao)方(fang)法。Class 對(dui)象(xiang)是在加載(zai)類時由(you) Java 虛擬機以及通過調(diao)用類加載(zai)器中(zhong)的 defineClass 方(fang)法自動構(gou)造(zao)的,因此不能顯式地聲明一(yi)個Class對(dui)象(xiang)。
虛(xu)擬(ni)機為每種類(lei)型(xing)管理一(yi)(yi)(yi)個獨一(yi)(yi)(yi)無(wu)二(er)的Class對象(xiang)(xiang)。也(ye)就是說,每個類(lei)(型(xing))都(dou)有一(yi)(yi)(yi)個Class對象(xiang)(xiang)。運(yun)行程序時,Java虛(xu)擬(ni)機(JVM)首先檢查是否所要加(jia)載(zai)的類(lei)對應的Class對象(xiang)(xiang)是否已經加(jia)載(zai)。如(ru)果沒有加(jia)載(zai),JVM就會根據類(lei)名查找.class文件(jian),并(bing)將其Class對象(xiang)(xiang)載(zai)入(ru)。
基本的 Java 類型(boolean、byte、char、short、int、long、float 和(he) double)和(he)關鍵字 void 也(ye)都對應一個(ge) Class 對象。
每個數組屬(shu)于被(bei)映射為 Class 對象的一個類,所有(you)具有(you)相同元(yuan)素類型和(he)維數的數組都共享該 Class 對象。
一般某個(ge)類的Class對象被載入內存,它就用來創(chuang)建(jian)這個(ge)類的所有對象。
一、如何得到Class的對象呢?
有三種方(fang)法可以的獲取:
1、調用(yong)Object類的(de)getClass()方(fang)法(fa)(fa)來得(de)到Class對象,這(zhe)也是常見的(de)產生Class對象的(de)方(fang)法(fa)(fa)。例如:
MyObject x;
Class c1 = x.getClass();
2、使用(yong)Class類中的靜態forName()方法(fa)獲得(de)與字符(fu)串對應(ying)的Class對象。例如(ru):
Class c2=Class.forName("Employee"),Employee必須(xu)是(shi)接口或者類的名字。
3、獲取Class類型對(dui)象(xiang)的(de)第(di)三(san)個方法非常簡(jian)單。如果T是一個Java類型,那么T.class就(jiu)代表(biao)了匹配的(de)類對(dui)象(xiang)。例(li)如
Class cl1 = Manager.class;
Class cl2 = int.class;
Class cl3 = Double[].class;
注意(yi):Class對象(xiang)實際(ji)上(shang)描(miao)述(shu)的(de)只是類(lei)型(xing),而這類(lei)型(xing)未必(bi)是類(lei)或者(zhe)接口(kou)。例如上(shang)面(mian)的(de)int.class是一個Class類(lei)型(xing)的(de)對象(xiang)。由于歷(li)史原因,數組類(lei)型(xing)的(de)getName方法會返回奇怪(guai)的(de)名字。
二、Class類的常用方法
1、getName()
一個(ge)Class對象(xiang)描述了一個(ge)特定類的(de)(de)屬性,Class類中常(chang)用的(de)(de)方法getName以 String 的(de)(de)形式返回此 Class 對象(xiang)所表示的(de)(de)實體(類、接口、數組類、基(ji)本類型或(huo) void)名稱(cheng)。
2、newInstance()
Class還(huan)有一個有用的方(fang)法(fa)可以(yi)為類創建一個實例(li),這個方(fang)法(fa)叫(jiao)做(zuo)newInstance()。例(li)如:
x.getClass.newInstance(),創建(jian)(jian)了(le)一個同x一樣類(lei)型的新(xin)實例。newInstance()方法(fa)調用默認(ren)構(gou)造(zao)器(無參數構(gou)造(zao)器)初(chu)始化新(xin)建(jian)(jian)對象。
3、getClassLoader()
返回該類(lei)的類(lei)加載器。
4、getComponentType()
返回(hui)表示數組(zu)組(zu)件類型的 Class。
5、getSuperclass()
返回(hui)表示此 Class 所表示的(de)實體(ti)(類(lei)、接口、基本類(lei)型或 void)的(de)超類(lei)的(de) Class。
6、isArray()
判定此 Class 對(dui)象是否表示一(yi)個數(shu)組類。
三、Class的一些使用技巧
1、forName和newInstance結合起來使用,可(ke)以(yi)根據存儲在字(zi)符串中(zhong)的類名創(chuang)建對(dui)象(xiang)。例(li)如
Object obj = Class.forName(s).newInstance();
2、虛擬機為每(mei)種類型管理一(yi)個獨一(yi)無二的(de)Class對(dui)(dui)象(xiang)。因此可以使用==操作符來比較類對(dui)(dui)象(xiang)。例(li)如:
if(e.getClass() == Employee.class)...
四.典型例題
1.將所(suo)有String類型的成員(yuan)變量(liang)里(li)的b改成a。
1 import java.lang.reflect.Field;
2 public class TestReflect {
3 public static void main(String[] args) throws SecurityException, NoSuchMethodException, NoSuchFieldException, IllegalArgumentException, Exception {
4 ReflectPointer rp1 = new ReflectPointer(3,4);
5 changeBtoA(rp1);
6 System.out.println(rp1);
7
8 }
9
10 private static void changeBtoA(Object obj) throws RuntimeException, Exception {
11 Field[] fields = obj.getClass().getFields();
12
13 for(Field field : fields) {
14 //if(field.getType().equals(String.class))
15 //由于字節(jie)碼(ma)只有一(yi)份,用equals語義不準確(que)
16 if(field.getType()==String.class) {
17 String oldValue = (String)field.get(obj);
18 String newValue = oldValue.replace('b', 'a');
19 field.set(obj,newValue);
20 }
21 }
22 }
23 }
24
25 class ReflectPointer {
26
27 private int x = 0;
28 public int y = 0;
29 public String str1 = "ball";
30 public String str2 = "basketball";
31 public String str3 = "itcat";
32
33 public ReflectPointer(int x,int y) {//alt + shift +s相當于右(you)鍵source
34 super();
35 // TODO Auto-generated constructor stub
36 this.x = x;
37 this.y = y;
38 }
39
40 @Override
41 public String toString() {
42 return "ReflectPointer [str1=" + str1 + ", str2=" + str2 + ", str3="
43 + str3 + "]";
44 }
45 }