当前位置: 首页 > article >正文

使用反射调用类的私有内部类的私有方法

文章目录

    • 使用反射调用类的私有方法
      • 实现方法
      • 实现代码
    • 使用反射调用类的私有内部类的私有方法
      • 实现方法
      • 实现代码

在进行单元测试时,我们往往需要直接访问某个类的内部类或者某个类的私有方法,此时正常的调用就无能为力了,因此我们可以使用反射进行调用。

使用反射调用类的私有方法

package net.mooctest;

public class outClass{
    public outClass() {

    }
    private void testMethod1(){
        System.out.println("调用了outClass的私有方法testMethod1");
    }
    
    //带参数的方法
    private void testMethod2(int a){
        System.out.println("调用了outClass的私有方法testMethod2,传入参数为:" + a);
    }
}

实现方法

  • 获取类的Class对象
    • Class class = Class.forName("类名")
    • Class class = 类实例.getClass()
  • new一个实例
    • Object object = class.newInstance();
  • 获取私有方法
    • Method method = class.getDeclaredMethod("方法名",参数类型1.class,参数类型2.class,...);
  • 设置方法可访问
    • method.setAccessible(true);
  • 调用私有方法
    • method.invoke(类实例,参数1,参数2,...);

实现代码


import static org.junit.jupiter.api.Assertions.*;

import java.lang.reflect.Method;

import org.junit.jupiter.api.Test;

import net.mooctest.outClass;

class outClassTest {

	@Test
	final void test1() throws Exception {
		outClass out = new outClass();
		
		Class class1 = out.getClass();
		
		Object object = class1.newInstance();
		//获取私有方法
		Method method = class1.getDeclaredMethod("testMethod1");
		//设置方法可访问
		method.setAccessible(true);
		//调用方法
		method.invoke(object);
		
		
	}

	@Test
	final void test2() throws Exception {
		outClass out = new outClass();
		Class class1 = out.getClass();
		Object object = class1.newInstance();
		//获取有参私有方法
		Method method = class1.getDeclaredMethod("testMethod2",int.class);
		//设置方法可访问
		method.setAccessible(true);
		//调用方法
		method.invoke(object,10);
			
	}

}

使用反射调用类的私有内部类的私有方法

package net.mooctest;

public class outClass{
    public outClass() {

    }
    private void testMethod1(){
        System.out.println("调用了outClass的私有方法testMethod1()");
    }
    
    //带参数的方法
    private void testMethod2(int a){
        System.out.println("调用了outClass的私有方法testMethod2(),传入参数为:" + a);
    }
    private class inClass{
    	public inClass() {
			// TODO Auto-generated constructor stub
		}
    	
    	public inClass(int a) {
			System.out.println("调用了有参构造器 " + a);
		}
    	private void inMethod1() {
    		System.out.println("调用了类的私有内部类的私有方法inMethod1()");
    	}
    }
}

实现方法

  • 获取私有内部类的Class对象
    • Class class = Class.forName("外部类名$内部类名")
  • 获取私有内部类构造器,无参构造器参数只填外部类名,有参构造器为(外部类名 ,参数类名…)
    • Constructor constructor = class.getDeclaredConstructor(outClass.class,参数类型1.class,参数类型2.class,...);
  • 设置构造器可访问
    • constructor.setAccessible(true);
  • new一个外部类对象,以便产生内部类实例(outClass:外部类名)
    • outClass out = new outClass();
  • 新建一个内部类实例,参数为外部类实例,若为有参构造器,后面再添加参数
    • Object object = constructor.newInstance(外部类实例,构造器参数1.class,构造器参数2.class,...);
  • 获取私有方法
    • Method method = class.getDeclaredMethod("inMethod1");
  • 设置方法可访问
    • method.setAccessible(true);
  • 调用方法
    • method.invoke(object);

实现代码

package test;

import static org.junit.jupiter.api.Assertions.*;

import java.lang.reflect.Constructor;
import java.lang.reflect.Method;

import org.junit.jupiter.api.Test;

import net.mooctest.outClass;

class outClassTest {
	@Test
	final void test3() throws Exception {
		
		//设置类名,外部类名$内部类名
		String className = "net.mooctest.outClass$inClass";
		Class class1 = Class.forName(className);
		//获取私有内部类构造器,无参构造器参数只填外部类名,有参构造器为(外部类名 ,参数类名...)
		Constructor constructor = class1.getDeclaredConstructor(outClass.class);
		
		//设置构造器可访问
		constructor.setAccessible(true);
		
		//new一个外部类对象,以便产生内部类实例
		outClass out = new outClass();
		//新建一个内部类实例,参数为外部类实例,若为有参构造器,后面再添加参数
		Object object = constructor.newInstance(out);
		//获取私有方法
		Method method = class1.getDeclaredMethod("inMethod1");
		//设置方法可访问
		method.setAccessible(true);
		//调用方法
		method.invoke(object);
			
	}
	
	@Test
	final void test4() throws Exception {
		
		//设置类名,外部类名$内部类名
		String className = "net.mooctest.outClass$inClass";
		Class class1 = Class.forName(className);
		//获取私有内部类构造器,无参构造器参数只填外部类名,有参构造器为(外部类名 ,参数类名...)
		Constructor constructor = class1.getDeclaredConstructor(outClass.class,int.class);
		
		//设置构造器可访问
		constructor.setAccessible(true);
		
		//new一个外部类对象,以便产生内部类实例
		outClass out = new outClass();
		//新建一个内部类实例,参数为外部类实例,若为有参构造器,后面再添加参数
		Object object = constructor.newInstance(out,10);
		//获取私有方法
		Method method = class1.getDeclaredMethod("inMethod1");
		//设置方法可访问
		method.setAccessible(true);
		//调用方法
		method.invoke(object);		
	}


}

http://www.kler.cn/news/108038.html

相关文章:

  • lvs+keepalived: 高可用集群
  • PHP 同城服务共享茶室小程序系统是如何实现的?
  • C语言二、八、十六进制转换
  • 【Apache Flink】实现有状态函数
  • 2023年马丁·加德纳聚会数学魔术分享之《不只是奇偶性的魔术》回顾
  • Android 13.0 根据包名授权悬浮窗权限
  • CSS宽度100%和宽度100vw之间有什么不同?
  • windows PC virtualBox 配置
  • 【计算机网络笔记】Web缓存/代理服务器技术
  • 用Visual Studio(VS)开发UNIX/Linux项目
  • excel巧用拼接函数CONCAT输出JSON、SQL字符串
  • Powershell脚本自动备份dhcp数据库
  • 僵尸网络|让人防不胜防的内部网络安全问题,作为企业IT不得不了解的基础安全
  • 酷开科技 | 酷开系统大屏电视,打造精彩家庭场景
  • Blueprints - 虚幻中的行为树(Behavior Tree)
  • 排序算法-堆积树排序法(HeapSort)
  • SQL企业微信群机器人消息推送
  • 关于数据中台的理解和思考
  • 实战经验分享FastAPI 是什么
  • Flutter笔记:完全基于Flutter绘图技术绘制一个精美的Dash图标(上)
  • react-antd组件 input输入框: 实现按回车搜索
  • 密码学与网络安全:量子计算的威胁与解决方案
  • 038-第三代软件开发-简易视频播放器-自定义Slider (二)
  • java后端返回数据给前端时去除值为空或NULL的属性、忽略某些属性
  • 聚观早报 |2024款飞凡R7官宣;小米14新配色材质
  • Spark新特性与核心概念
  • 网络(番外篇)can网络知识
  • VScode 调试 linux内核
  • 【错误解决方案】ModuleNotFoundError: No module named ‘cPickle‘
  • SQL Server Management Studio (SSMS)的安装教程