1. 程式人生 > 實用技巧 >類載入器,,,,,反射,,,,,泛型擦除,,,,, 反射配置檔案

類載入器,,,,,反射,,,,,泛型擦除,,,,, 反射配置檔案

1.1類的載入

當程式要使用某個類時,如果該類還未被載入到記憶體中,則系統會通過載入,連線,初始化三步來實現對這個類進行初始化。

l載入

就是指將class檔案讀入記憶體,併為之建立一個Class物件。

任何類被使用時系統都會建立一個Class物件

l連線

驗證 是否有正確的內部結構,並和其他類協調一致

準備 負責為類的靜態成員分配記憶體,並設定預設初始化值

解析 將類的二進位制資料中的符號引用替換為直接引用

l初始化

就是我們以前講過的初始化步驟

1.1類初始化時機

1. 建立類的例項

2. 類的靜態變數,或者為靜態變數賦值

3. 類的靜態方法

4.使用反射方式來強制建立某個類或介面對應的java.lang.Class物件

5. 初始化某個類的子類

6. 直接使用java.exe命令來執行某個主類

1.1 反射

1.1Class

l獲取Class物件的三種方式

方式: 通過Object類中的getObject()方法

Person p = new Person();

Class c = p.getClass();

方式二:通過 類名.class 獲取到位元組碼檔案物件(任意資料型別都具備一個class靜態屬性,看上去要比第一種方式簡單)。

Class c2 = Person.class;

方式三:通過Class類中的方法(將類名作為字串傳遞給Class類中的靜態方法forName即可)。

Class c3 = Class.forName("Person");

l注意:第三種和前兩種的區別

前兩種你必須明確Person型別.

後面是指定這種型別的字串就行.這種擴充套件更強.我不需要知道你的類.我只提供字串,按照配置檔案載入就可以了。

l程式碼演示

 * 獲取.class位元組碼檔案物件的方式
 *         1:通過Object類中的getObject()方法
 *         2: 通過 類名.class 獲取到位元組碼檔案物件
 *         3: 反射中的方法,
 *             public static Class<?> forName(String className) throws ClassNotFoundException
 
* 返回與帶有給定字串名的類或介面相關聯的 Class 物件 */ public class ReflectDemo { public static void main(String[] args) throws ClassNotFoundException { // 1: 通過Object類中的getObject()方法 // Person p1 = new Person(); // Class c1 = p1.getClass(); // System.out.println("c1 = "+ c1); // 2: 通過 類名.class 獲取到位元組碼檔案物件 // Class c2 = Person.class; // System.out.println("c2 = "+ c2); // 3: 反射中的方法 Class c3 = Class.forName("cn.oracle_01_Reflect.Person");// 包名.類名 System.out.println("c3 = " + c3); } }

lPerson類

package cn.oracle_01_Reflect;
public class Person {
    //成員變數
    public String name;
    public int age;
    private String address;

    //構造方法
    public Person() {
        System.out.println("空引數構造方法");
    }
    
    public Person(String name) {
        this.name = name;
        System.out.println("帶有String的構造方法");
    }
    //私有的構造方法
    private Person(String name, int age){
        this.name = name;
        this.age = age;
        System.out.println("帶有String,int的構造方法");
    }
    
    public Person(String name, int age, String address){
        this.name = name;
        this.age = age;
        this.address = address;
        System.out.println("帶有String, int, String的構造方法");
    }
    
    //成員方法
    //沒有返回值沒有引數的方法
    public void method1(){
        System.out.println("沒有返回值沒有引數的方法");
    }
    //沒有返回值,有引數的方法
    public void method2(String name){
        System.out.println("沒有返回值,有引數的方法 name= "+ name);
    }
    //有返回值,沒有引數
    public int method3(){
        System.out.println("有返回值,沒有引數的方法");
        return 123;
    }
    //有返回值,有引數的方法
    public String method4(String name){
        System.out.println("有返回值,有引數的方法");
        return "哈哈" + name;
    }
    //私有方法
    private void method5(){
        System.out.println("私有方法");
    }

    @Override
    public String toString() {
        return "Person [name=" + name + ", age=" + age + ", address=" + address+ "]";
    }
}

在反射機制中,把類中的成員(構造方法、成員方法、成員變數)都封裝成了對應的類進行表示。其中,構造方法使用類Constructor表示。可通過Class類中提供的方法獲取構造方法

l返回一個構造方法

npublic Constructor<T> getConstructor(Class<?>... parameterTypes) 獲取public修飾, 指定引數型別所對應的構造方法

npublic Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes) 獲取指定引數型別所對應的構造方法(包含私有的)

l返回多個構造方法

npublic Constructor<?>[] getConstructors() 獲取所有的public 修飾的構造方法

npublic Constructor<?>[] getDeclaredConstructors() 獲取所有的構造方法(包含私有的)

public class ReflectDemo {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, SecurityException {
        //獲取Class物件
        Class c = Class.forName("cn.oracle_01_Reflect.Person");//包名.類名
        
        //獲取所有的構造方法
        //Constructor[] cons = c.getConstructors();
        Constructor[] cons = c.getDeclaredConstructors();
        for (Constructor con : cons) {
            System.out.println(con);
        }
        
        System.out.println("------------------------");
        //獲取一個構造方法
        //public Person() 
        Constructor con1 = c.getConstructor(null);
        System.out.println(con1);
        
        //public Person(String name)
        Constructor con2 = c.getConstructor(String.class);
        System.out.println(con2);
        
        //private Person(String name, int age)
        Constructor con3 = c.getDeclaredConstructor(String.class, int.class);
        System.out.println(con3);
        
        //public Person(String name, int age, String address)
        Constructor con4 = c.getDeclaredConstructor(String.class, int.class, String.class);
        System.out.println(con4);
    }
}

通過反射獲取成員變數並使用

在反射機制中,把類中的成員變數使用類Field表示。可通過Class類中提供的方法獲取成員變數:

l 返回一個成員變數

public Field getField(String name) 獲取指定的 public修飾的變數

public Field getDeclaredField(String name) 獲取指定的任意變數

l 返回多個成員變數

public Field[] getFields() 獲取所有public 修飾的變數

public Field[] getDeclaredFields() 獲取所有的 變數 (包含私有)

//通過反射獲取成員變數並使用
public class Demo01 {
public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, SecurityException, InstantiationException, IllegalAccessException {
    //獲取Person類的位元組碼檔案
    Class c=Class.forName("com.oracle.demo06.Person");
    //獲取成員變數物件
    Field fl=c.getField("name");
    //System.out.println(fl);
    //快速建立物件,只能呼叫空參構造
    Person p=(Person)c.newInstance();
    fl.set(p,"小紅帽");
    System.out.println(p.name);//獲得成員變數
}
}

泛型擦除

程式編譯後產生的.class檔案中是沒有泛型約束的,這種現象我們稱為泛型的擦除

//泛型擦除,泛型不進class檔案
public class Demo03 {
public static void main(String[] args) throws NoSuchMethodException, SecurityException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
    ArrayList<String> arr=new ArrayList<String>();
    arr.add("abc");
    //獲取arr的位元組碼檔案物件
    Class c=arr.getClass();
    //獲取add的方法物件
    Method method=c.getMethod("add", Object.class);
    //呼叫方法:執行add()方法
    method.invoke(arr, 123);
    System.out.println(arr);
}
}

反射配置檔案

通過反射配置檔案,執行配置檔案中指定類的對應方法

讀取Peoperties.txt檔案中的資料,通過反射技術,來完成Person物件的建立

package com.oracle.demo08;

public class Worker {

    public void work(){
        System.out.println("工人工作");
    
}
}
 

public class Student {
public void study(){
    System.out.println("學生學習");
}
}
public class Doctor {

    public void job(){
        System.out.println("醫生工作");
    
}
}
 

public class Demo01 {
public static void main(String[] args) throws IOException, ClassNotFoundException, NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
    //建立Properties集合物件
    Properties pro=new Properties();
    //建立輸入流讀取
    FileInputStream fis=new FileInputStream("src/com/oracle/demo08/pro.properties");
    pro.load(fis);//讀到集合中
    //獲取類名
    String className=pro.getProperty("className");
    String methodName=pro.getProperty("methodName");
    //反射獲取位元組碼檔案
    Class c=Class.forName(className);
    //獲取方法物件
    Method m=c.getMethod(methodName);
    //快速建立物件
    Object obj=c.newInstance();
    //呼叫方法
    m.invoke(obj);
    
}
}

建立物件的三種方法:new,反序列,反射

package cn.oracle_01_Reflect;

publicclassPerson {

//成員變數

publicString name;

publicintage;

privateString address;

//構造方法

publicPerson() {

System.out.println("空引數構造方法");

}

publicPerson(String name) {

this.name= name;

System.out.println("帶有String的構造方法");

}

//私有的構造方法

privatePerson(String name, intage){

this.name= name;

this.age= age;

System.out.println("帶有Stringint的構造方法");

}

publicPerson(String name, intage, String address){

this.name= name;

this.age= age;

this.address= address;

System.out.println("帶有String, int, String的構造方法");

}

//成員方法

//沒有返回值沒有引數的方法

publicvoidmethod1(){

System.out.println("沒有返回值沒有引數的方法");

}

//沒有返回值,有引數的方法

publicvoidmethod2(String name){

System.out.println("沒有返回值,有引數的方法 name= "+ name);

}

//有返回值,沒有引數

publicintmethod3(){

System.out.println("有返回值,沒有引數的方法");

return123;

}

//有返回值,有引數的方法

publicString method4(String name){

System.out.println("有返回值,有引數的方法");

return"哈哈"+ name;

}

//私有方法

privatevoidmethod5(){

System.out.println("私有方法");

}

@Override

publicString toString() {

return"Person [name="+ name+ ", age="+ age+ ", address="+ address+ "]";

}

}