中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java反射的運用方法實例分析

發布時間:2022-03-09 09:14:18 來源:億速云 閱讀:137 作者:iii 欄目:開發技術

今天小編給大家分享一下Java反射的運用方法實例分析的相關知識點,內容詳細,邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。

    反射

    反射定義

    對象可以通過反射獲取他的類,類可以通過反射拿到所有?法(包括私有) 通過java語言中的反射機制可以操作字節碼文件,可以讀和修改字節碼文件

    反射的基本運用

    1. 獲取類對象

    a. forName()方法

    只需要知道類名,在加載JDBC的時候會采用 實例代碼

    public class test1 {
        public static void main(String[] args) throws ClassNotFoundException {
            Class name = Class.forName("java.lang.Runtime");
            System.out.println(name);
        }
    }

    Java反射的運用方法實例分析

    b. 直接獲取

    使用.class去獲取對于的對象

    public class test1 {
        public static void main(String[] args) throws ClassNotFoundException {
            Class<?> name = Runtime.class;
            System.out.println(name);
        }
    }
    c. getClass()方法

    getClass來獲取字節碼對象,必須要明確具體的類,然后創建對象

    public class test1 {
        public static void main(String[] args) throws ClassNotFoundException {
            Runtime rt = Runtime.getRuntime();
            Class<?> name = rt.getClass();
            System.out.println(name);
        }
    }
    d. getSystemClassLoader().loadClass()方法

    這個方法和forName類似,只要有類名就可以了,但是區別在于,forName的靜態JVM會裝載類,并執行static()中的代碼

    public class getSystemClassLoader {
        public static void main(String[] args) throws ClassNotFoundException {
            Class<?> name = ClassLoader.getSystemClassLoader().loadClass("java.lang.Runtime");
            System.out.println(name);
        }
    }

    2. 獲取類方法

    a. getDeclaredMethods

    返回類或接口聲明的所有方法,包括public、protected、private和默認方法,但是不包括繼承的方法

    import java.lang.reflect.Method;
    
    public class getDeclaredMethods {
        public static void main(String[] args) throws ClassNotFoundException {
            Class<?> name = Class.forName("java.lang.Runtime");
            System.out.println(name);
            Method[] m = name.getDeclaredMethods();
            for(Method x:m)
                System.out.println(x);
        }
    }

    Java反射的運用方法實例分析

    b. getDeclaredMethod

    獲取特定的方法,第一個參數是方法名,第二個參數是該方法的參數對應的class對象,例如這里Runtime的exec方法參數為一個String,所以這里的第二個參數是String.class

    import java.lang.reflect.Method;
    
    public class getDeclaredMethod {
        public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException {
            Class<?> name = Class.forName("java.lang.Runtime");
            Method m = name.getDeclaredMethod("exec",String.class);
            System.out.println(m);
        }
    }
    c. getMethods

    返回某個類所有的public方法,包括繼承類的public方法

    d. getMethod

    參數同理getDeclaredMethod

    3. 獲取成員變量

    同理Method的那幾個方法

    a. getDeclaredFields

    獲取類的成員的所有變量數組,但是不包括父類的

    b. getDeclaredField(String name)

    獲取特定的,參數是想要的方法的名稱

    c. getFields()

    同理,只能獲得public的,但是包括了父類的

    d. getField(String name)

    同理,參數是想要的方法的名稱

    4. 獲取構造函數Constructor

    Constructor<?>[] getConstructors() :只返回public構造函數

    Constructor<?>[] getDeclaredConstructors() :返回所有構造函數

    Constructor<> getConstructor(類<?>... parameterTypes) : 匹配和參數配型相符的public構造函數

    Constructor<> getDeclaredConstructor(類<?>... parameterTypes) : 匹配和參數配型相符的構造函數

    后面兩個方法的參數是對于方法的參數的類型的class對象,和Method的那個類似,例如String.class

    5. 反射創建類對象

    newInstance

    可以通過反射來生成實例化對象,一般我們使用Class對象的newInstance()方法來進行創建類對象

    創建的方法就是:只需要通過forname方法獲取到的class對象中進行newInstance方法創建即可

    Class c = Class.forName("com.reflect.MethodTest"); // 創建Class對象
    Object m1 =  c.newInstance(); // 創建類對象
    invoke

    invoke方法位于java.lang.reflect.Method類中,用于執行某個的對象的目標方法,一般會和getMethod方法配合進行調用。

    使用用法:

    public Object invoke(Object obj, Object... args)

    第一個參數為類的實例,第二個參數為相應函數中的參數

    obj:從中調用底層方法的對象,必須是實例化對象

    args: 用于方法的調用,是一個object的數組,參數有可能是多個

    但需要注意的是,invoke方法第一個參數并不是固定的:

    • 如果調用這個方法是普通方法,第一個參數就是類對象;

    • 如果調用這個方法是靜態方法,第一個參數就是類;

    通過一個例子去理解

    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method;
    
    public class Invoke {
        public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
            Class c = Class.forName("Invoke");
            Object o = c.newInstance();
            Method m = c.getMethod("test");
            m.invoke(o);
        }
        public void test(){
            System.out.println("測試成功");
        }
    }

    Java反射的運用方法實例分析

    簡單來說就是這樣

    方法.invoke(類或類對象)

    先forName拿到Class,再newInstance獲取類對象,再getMethod獲取方法,然后調用

    Runtime的rce例子(訪問限制突破)

    Runtime類里面有一個exec方法,可以執行命令

    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method;
    
    public class Exec {
        public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
            Class c = Class.forName("java.lang.Runtime");
            Object o = c.newInstance();
            Method m = c.getMethod("exec",String.class);
            m.invoke(o,"/System/Applications/Calculator.app/Contents/MacOS/Calculator");
        }
    }

    但是發現報錯了

    Java反射的運用方法實例分析

    出現這個問題的原因:

    • 使用的類沒有無參構造函數

    • 使用的類構造函數是私有的

    那么解決方案就是setAccessible(true);,用這個去突破訪問限制

    Java.lang.reflect.AccessibleObject類是Field,Method和Constructor類對象的基類,可以提供將反射對象標記為使用它抑制摸人Java訪問控制檢查的功能,同時上述的反射類中的Field,Method和Constructor繼承自AccessibleObject。所以我們在這些類方法基礎上調用setAccessible()方法,既可對這些私有字段進行操作

    簡單來說,私有的屬性、方法、構造方法,可以通過這個去突破限制,xxx.setAccessible(true) 可以看到Runtime的構造方法是private的

    Java反射的運用方法實例分析

    那么這里我們就可以這么去突破限制 先獲取構造方法,然后setAccessible獲取訪問權限 然后再最后invoke里面,第一個參數寫成con.newInstance()

    import java.lang.reflect.Constructor;
    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method;
    
    public class Exec {
        public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, InstantiationException {
            Class c = Class.forName("java.lang.Runtime");
            Constructor con = c.getDeclaredConstructor();
            con.setAccessible(true);
            Method m = c.getMethod("exec",String.class);
            m.invoke(con.newInstance(),"/System/Applications/Calculator.app/Contents/MacOS/Calculator");
        }
    }

    Java反射的運用方法實例分析

    這里有一個疑問,如果把con.newInstance單獨提取出來,他打開計算器不會顯示出來,但是后臺的確是啟動了,不知道啥原因

    import java.lang.reflect.Constructor;
    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method;
    
    public class Exec {
        public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException, InstantiationException {
            Class c = Class.forName("java.lang.Runtime");
            Constructor con = c.getDeclaredConstructor();
            con.setAccessible(true);
            Object o = con.newInstance();
            Method m = c.getMethod("exec",String.class);
            m.invoke(o,"/System/Applications/Calculator.app/Contents/MacOS/Calculator");
        }
    }

    后記

    反射中常用的幾個重要方法:

    • 獲取類的?法: forName

    • 實例化類對象的?法: newInstance

    • 獲取函數的?法: getMethod

    • 執?函數的?法: invoke

    • 限制突破方法:setAccessible

    以上就是“Java反射的運用方法實例分析”這篇文章的所有內容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學習更多的知識,請關注億速云行業資訊頻道。

    向AI問一下細節

    免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

    AI

    四川省| 拜泉县| 天长市| 西峡县| 英德市| 普陀区| 云龙县| 吉安县| 会宁县| 蒲城县| 台州市| 思茅市| 巴彦淖尔市| 清苑县| 神农架林区| 本溪| 北宁市| 隆安县| 墨竹工卡县| 广饶县| 灌阳县| 城口县| 富宁县| 阆中市| 兴宁市| 当阳市| 井冈山市| 和平县| 明光市| 盱眙县| 栾城县| 中西区| 正宁县| 封开县| 依安县| 万荣县| 文山县| 锡林浩特市| 武功县| 新巴尔虎右旗| 邢台市|