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

news/2024/5/19 3:31:06 标签: java, 反射, 私有内部类

文章目录

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

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

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

java">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,...);

实现代码

java">
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);
			
	}

}

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

java">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);

实现代码

java">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.niftyadmin.cn/n/5130682.html

相关文章

口袋参谋:如何快速挑选宝贝核心关键词?三种方法,简单有效!

做电商,一定要会选择关键词!这是我近十年做电商的经验之谈。 不管是标题还是直通车推广,都需要选择与产品相关度高、符合买家搜索习惯的关键词,只有这样,才能吸引更多自然流量!有了流量加持,还…

Cyanine5 Alkyne在生物分子标记与追踪中的应用

Cyanine5 Alkyne作为一种荧光探针在生物分子标记与追踪中具有诸多应用。其炔基官能团赋予了它在点击化学反应中的特异性,使得它能够与含有偶极炉官能团的生物分子发生共轭加成反应,从而实现对特定生物分子的标记与追踪。以下是Cyanine5 Alkyne在生物分子…

继续畅通工程(最小生成树+并查集)

我刚开始的想法是,既然有已经修好的路,就先把这些修好的路全都加到集合中,即直接把他们Union,不加其长度,然后再将剩下的排序,按Kruskal的算法来进行。 但是这种想法可能有什么问题,只过了一般…

算法训练营第三天 | 203.移除链表元素、707.设计链表 、206.反转链表

关于链表我们应该了解什么: 代码随想录 在实际开发中,遇到指针我们要做好防御性编程。 问题( 一 ) 题目描述 : 给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val val 的节点…

正确部署Baichuan2(Ubuntu20.4) 步骤及可能出现的问题

部署其实是不太复杂的,但实际上也耗费了接近2-3天的时间去不断的设置 1 硬件配置信息 采用esxi 虚拟化的方式将T4 卡穿透给esxi 种的ubuntu20.4虚拟机 CPU给到8 core 内存至少32GB以上 T4卡是16GB 2 预先准备OS环境 这里使用的是ubuntu20.4版本,esxi中需要设置uefI启动方…

【Python机器学习】零基础掌握check_increasing等渗回归

有没有想过为什么在线购物推荐系统总能准确地推荐出喜欢的商品? 在当今的大数据时代,推荐系统无处不在,从电商网站到社交媒体,都在用复杂的算法来预测喜好和趋势。但这其中有一个不为人知的问题:如何确保推荐结果是单调递增或递减的,即保证推荐结果与用户的实际需求更加…

【性能测试】初识 Jmeter 中的 BeanShell

初识 Jmeter 中的 BeanShell 1.简介1.1 应用场景1.2 BeanShell 类型 2.常用内置变量2.1 log 日志模块2.2 vars 模块2.3 props 模块2.4 prev 模块 3.常见应用场景3.1 Java 文件处理3.2 导入外部 jar 包 BeanShell 是一个小型嵌入式 Java 源代码解释器,完全兼容 Java …

中国黑客群体的收入,与国外的黑客调查问卷相比!竟然还有女黑客!

从圈外认知来说,黑客一直被认为是高收入群体,黑客有白帽和黑帽处于黑白两道的黑客会的技术都有些相似,但是却是对立的,白帽做网络安全,修补漏洞。黑帽各种破坏,挖数据,攻击漏洞。 如果你对网络…