欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

一文詳解Java中的反射與new創(chuàng)建對象

 更新時間:2024年07月18日 09:50:44   作者:碼農(nóng)研究僧  
Java中的反射(Reflection)和使用new關鍵字創(chuàng)建對象是兩種不同的對象創(chuàng)建方式,各有優(yōu)缺點和適用場景,本文小編給大家詳細介紹了Java中的反射與new創(chuàng)建對象,感興趣的小伙伴跟著小編一起來看看吧

1. 基本知識

Java中的反射(Reflection)和使用new關鍵字創(chuàng)建對象是兩種不同的對象創(chuàng)建方式,各有優(yōu)缺點和適用場景

第一:使用new關鍵字創(chuàng)建對象

優(yōu)點

  • 編譯時類型檢查:在編譯時就可以檢查類型是否正確
  • 性能高:不涉及額外的解析和方法調(diào)用,直接創(chuàng)建對象

缺點

  • 靈活性差:不能在運行時動態(tài)決定要創(chuàng)建的對象類型

基本的測試代碼如下:

public class Main {
    public static void main(String[] args) {
        // 使用new關鍵字創(chuàng)建對象
        MyClass obj = new MyClass();
        obj.sayHello();
    }
}

class MyClass {
    public void sayHello() {
        System.out.println("Hello, World!");
    }
}

二、使用反射創(chuàng)建對象

優(yōu)點

  • 靈活性高:可以在運行時動態(tài)決定要創(chuàng)建的對象類型,適用于框架和工具類

缺點

  • 性能較低:由于需要解析類名和方法名,性能比直接使用new關鍵字低
  • 缺乏編譯時類型檢查:可能會在運行時拋出異常

示例代碼如下:

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

public class Main {
    public static void main(String[] args) {
        try {
            // 使用反射創(chuàng)建對象
            Class<?> clazz = Class.forName("MyClass");
            Constructor<?> constructor = clazz.getConstructor();
            MyClass obj = (MyClass) constructor.newInstance();
            obj.sayHello();
        } catch (ClassNotFoundException | NoSuchMethodException | InstantiationException | IllegalAccessException | InvocationTargetException e) {
            e.printStackTrace();
        }
    }
}

class MyClass {
    public void sayHello() {
        System.out.println("Hello, World!");
    }
}

簡單分析其效率對比 以及 使用的場景

方面使用new關鍵字使用反射
創(chuàng)建對象的效率高效:編譯后生成的字節(jié)碼直接創(chuàng)建對象,無需額外解析和方法調(diào)用較低:需要解析類名和方法名、獲取構(gòu)造器等操作,額外增加了開銷
方法調(diào)用的效率高效:直接調(diào)用對象方法,無額外開銷較低:需通過反射API調(diào)用方法,效率較低
使用場景確定的場景下,直接創(chuàng)建對象,例如日常開發(fā)中的大部分情況需要在運行時動態(tài)決定要創(chuàng)建的對象類型,例如框架開發(fā)(如Spring框架中的依賴注入)、工具類開發(fā)(如ORM框架)

2. 效率對比

public class PerformanceTest {
    public static void main(String[] args) {
        final int iterations = 1000000;

        // 測試new關鍵字
        long startNew = System.nanoTime();
        for (int i = 0; i < iterations; i++) {
            MyClass obj = new MyClass();
        }
        long endNew = System.nanoTime();
        System.out.println("Using new keyword: " + (endNew - startNew) + " ns");

        // 測試反射
        long startReflection = System.nanoTime();
        try {
            for (int i = 0; i < iterations; i++) {
                Class<?> clazz = Class.forName("MyClass");
                Constructor<?> constructor = clazz.getConstructor();
                MyClass obj = (MyClass) constructor.newInstance();
            }
        } catch (ClassNotFoundException | NoSuchMethodException | InstantiationException | IllegalAccessException | InvocationTargetException e) {
            e.printStackTrace();
        }
        long endReflection = System.nanoTime();
        System.out.println("Using reflection: " + (endReflection - startReflection) + " ns");
    }
}

class MyClass {
    public void sayHello() {
        System.out.println("Hello, World!");
    }
}

截圖如下:

具體時間因機器性能和JVM狀態(tài)不同而異

3. 反射補充知識

反射創(chuàng)建對象的三種主要方式是通過類字面常量、Class.forName()方法和對象實例的getClass()方法

3.1 類字面場量

在編譯時就知道類的類型,適用于類名已知且在同一編譯單元中的情況

public class ReflectDemo {
    public void sayHello() {
        System.out.println("Hello, World!");
    }

    public static void main(String[] args) {
        // 方式1:使用類字面常量
        Class<ReflectDemo> reflectDemoClass = ReflectDemo.class;
        try {
            ReflectDemo reflectDemo = reflectDemoClass.newInstance();
            reflectDemo.sayHello();
        } catch (InstantiationException | IllegalAccessException e) {
            e.printStackTrace();
        }
    }
}

3.2 Class.forName()

適用于類名在運行時才知道的情況,常用于從配置文件或輸入中讀取類名,然后動態(tài)加載類

public class ReflectDemo {
    public void sayHello() {
        System.out.println("Hello, World!");
    }

    public static void main(String[] args) {
        try {
            // 方式2:使用Class.forName()
            Class<?> reflectDemoClass = Class.forName("ReflectDemo");
            ReflectDemo reflectDemo = (ReflectDemo) reflectDemoClass.newInstance();
            reflectDemo.sayHello();
        } catch (ClassNotFoundException | InstantiationException | IllegalAccessException e) {
            e.printStackTrace();
        }
    }
}

3.3 對象實例的 getClass()

適用于已有實例對象時,通過該對象獲取其類信息,然后創(chuàng)建新的實例

public class ReflectDemo {
    public void sayHello() {
        System.out.println("Hello, World!");
    }

    public static void main(String[] args) {
        ReflectDemo existingInstance = new ReflectDemo();
        // 方式3:使用對象實例的getClass()
        Class<? extends ReflectDemo> reflectDemoClass = existingInstance.getClass();
        try {
            ReflectDemo newReflectDemo = reflectDemoClass.newInstance();
            newReflectDemo.sayHello();
        } catch (InstantiationException | IllegalAccessException e) {
            e.printStackTrace();
        }
    }
}

3.4 總結(jié)

方式優(yōu)點缺點
使用類字面常量 (ClassName.class)- 編譯時類型安全
- 性能最佳
- 類名在編譯時必須已知
- 靈活性較差
使用 Class.forName()- 可以在運行時根據(jù)類名動態(tài)加載類
- 靈活性高
- 類名必須為字符串形式
- 可能會引發(fā)ClassNotFoundException
- 性能略低于類字面常量
使用對象實例的 getClass()- 可以通過已有實例獲取類信息
- 動態(tài)創(chuàng)建相同類型的新實例
- 需要先有一個實例對象
- 性能略低于類字面常量,但高于Class.forName()

以上就是一文詳解Java中的反射與new創(chuàng)建對象的詳細內(nèi)容,更多關于Java反射與new創(chuàng)建對象的資料請關注腳本之家其它相關文章!

相關文章

  • 使用Mybatis實現(xiàn)分頁效果示例

    使用Mybatis實現(xiàn)分頁效果示例

    大家好,本篇文章主要講的是使用Mybatis實現(xiàn)分頁效果示例,感興趣的同學趕快來看一看吧,對你有幫助的話記得收藏一下,方便下次瀏覽
    2021-12-12
  • 如何使用XPath提取xml文檔數(shù)據(jù)

    如何使用XPath提取xml文檔數(shù)據(jù)

    這篇文章主要介紹了如何使用XPath提取xml文檔數(shù)據(jù),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2019-08-08
  • 淺談Java中父類與子類的加載順序詳解

    淺談Java中父類與子類的加載順序詳解

    本篇文章是對Java中父類與子類的加載順序進行了詳細的分析介紹,需要的朋友參考下
    2013-06-06
  • Java語言class類用法及泛化(詳解)

    Java語言class類用法及泛化(詳解)

    這篇文章主要介紹了Java語言class類用法及泛化(詳解),大家都知道Java程序在運行過程中,對所有的對象今夕類型標識,也就是RTTI。這項信息記錄了每個對象所屬的類,需要的朋友可以參考下
    2015-07-07
  • Spring Boot的Maven插件Spring Boot Maven plugin詳解

    Spring Boot的Maven插件Spring Boot Maven plu

    Spring Boot的Maven插件Spring Boot Maven plugin以Maven的方式提供Spring Boot支持,Spring Boot Maven plugin將Spring Boot應用打包為可執(zhí)行的jar或war文件,然后以通常的方式運行Spring Boot應用,本文介紹Spring Boot的Maven插件Spring Boot Maven plugin,一起看看吧
    2024-01-01
  • Java中toString()、String.valueOf、(String)強轉(zhuǎn)區(qū)別

    Java中toString()、String.valueOf、(String)強轉(zhuǎn)區(qū)別

    相信大家在日常開發(fā)中這三種方法用到的應該很多,本文主要介紹了Java中toString()、String.valueOf、(String)強轉(zhuǎn)區(qū)別,感興趣的可以了解一下
    2021-09-09
  • java Timer 定時每天凌晨1點執(zhí)行任務

    java Timer 定時每天凌晨1點執(zhí)行任務

    這篇文章主要介紹了java Timer 定時每天凌晨1點執(zhí)行任務的代碼,代碼簡單易懂,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-09-09
  • java中Collections.sort排序詳解

    java中Collections.sort排序詳解

    這篇文章主要介紹了java中Collections.sort排序詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-12-12
  • spring.datasource.schema配置詳解

    spring.datasource.schema配置詳解

    本文主要介紹了spring.datasource.schema配置,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-05-05
  • Mybatis結(jié)果生成鍵值對的實例代碼

    Mybatis結(jié)果生成鍵值對的實例代碼

    這篇文章主要介紹了Mybatis結(jié)果生成鍵值對的實例代碼,以及MyBatis返回Map鍵值對數(shù)據(jù)的實現(xiàn)方法,非常不錯,具有參考借鑒價值,需要的的朋友參考下
    2017-02-02

最新評論