怎樣通過反射獲取非靜態內部類實例

一、無參構造函數

/**
 *  反射內部類
 * @author zhangyu
 * @date 2022/1/8
*/
public class OuterClass {

    public void process() {
        System.out.println("外部類方法...");
    }

    class InnerClass {
        public void process() {
            System.out.println("內部類方法...");
        }
    }
}

【1】獲取內部類對象需要依賴外部類的對象

【2】通過Class獲取內部類可以通過倆種方式

  • 第一種:Class.forName("package.name.OuterClass I n n e r C l a s s " ) 通 過 InnerClass") 通過 InnerClass")通過符號定位到內部類
  • 第二種:通過外部類Class使用OuterClass.class.getDeclaredClasses()獲取外部類下可用的內部類對象

【3】實例化類需要構造器,反射調用構造器需要設置可訪問constructor.setAccessible(true);

    @Test
    public  void test5() throws InvocationTargetException, InstantiationException, IllegalAccessException, NoSuchMethodException, ClassNotFoundException {
        //需要依賴外部類對象實例化內部類
        Class outerClass = Class.forName("reflection.demo.OuterClass");
        Object outerObject = outerClass.newInstance();
        //如果知道外部類下內部類數量,那麼通過下面方式即可
        Class<?> innerClass = OuterClass.class.getDeclaredClasses()[0];
        //如果知道內部類全限定名,可以通過下面的方式
        //Class<?> innerClass = Class.forName("full.package.name.OuterClass$InnerClass")
        //獲取默認構造器
        Constructor<?> constructor = innerClass.getDeclaredConstructors()[0];
        //也可以通過下面的方式獲取構造器
        //Constructor<?> constructor = innerClass.getDeclaredConstructor(OuterClass.class);
        //默認構造器私有的,需要設置可訪問
        constructor.setAccessible(true);
        //通過構造器和父類Class創建內部類實例
        Object innerObject = constructor.newInstance(outerObject);
        Method innerClassDeclaredMethod = innerClass.getDeclaredMethod("process");
        innerClassDeclaredMethod.setAccessible(true);
        innerClassDeclaredMethod.invoke(innerObject);
    }

二、有參構造函數

上面描述的是不帶參數的構造函數,下面描述下帶參數的構造函數

/**
 *  反射內部類
 * @author zhangyu
 * @date 2022/1/8
*/
public class OuterClass {

    private  String name;

    public OuterClass(String name) {
        this.name = name;
    }

    public void process() {
        System.out.println("外部類方法...");
    }

    class InnerClass {
        private  String name;

        public InnerClass(String name) {
            this.name = name;
        }

        public void process() {
            System.out.println(name+"----內部類方法...");
        }
    }
}
    @Test
    public  void test5() throws InvocationTargetException, InstantiationException, IllegalAccessException, NoSuchMethodException, ClassNotFoundException {
        //需要依賴外部類對象實例化內部類
        Class outerClass = Class.forName("reflection.demo.OuterClass");
        Constructor declaredConstructor = outerClass.getDeclaredConstructor(String.class);
        declaredConstructor.setAccessible(true);
        //獲取外部類實例以及設置構造函數值
        Object outerObject = declaredConstructor.newInstance("outerClass");

        //如果知道外部類下內部類數量,那麼通過下面方式即可
        Class<?> innerClass = OuterClass.class.getDeclaredClasses()[0];
        //如果知道內部類全限定名,可以通過下面的方式
        //Class<?> innerClass = Class.forName("full.package.name.OuterClass$InnerClass")
        //獲取默認構造器
        Constructor<?> constructor = innerClass.getDeclaredConstructors()[0];
        //也可以通過下面的方式獲取構造器
        //Constructor<?> constructor = innerClass.getDeclaredConstructor(OuterClass.class);
        //默認構造器私有的,需要設置可訪問
        constructor.setAccessible(true);
        //通過構造器和父類Class創建內部類實例.設置構造函數值
        Object innerObject = constructor.newInstance(outerObject,"innerClass");
        Method innerClassDeclaredMethod = innerClass.getDeclaredMethod("process");
        innerClassDeclaredMethod.setAccessible(true);
        innerClassDeclaredMethod.invoke(innerObject);
    }

總結

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。

推薦閱讀: