java通过反射获取不到内部类
一.场景
在配置线程池的时候,我们想把线程池的拒绝策略全类名(该类是一个内部类)通过Value注解从配置文件中注入到PoolExecutorConfig类的rejectedExecutionHandler字段,但是我们将该字段传入Class.forName方法的时候,抛出了ClassNotFoundException异常。
二.问题复现
1.1.线程池的四种拒绝策略
当线程池的任务缓存队列已满并且线程池中的线程数目达到maximumPoolSize时,如果还有任务到来就会采取任务拒绝策略,通常有以下四种策略:
- AbortPolicy:丢弃任务并抛出RejectedExecutionException异常
- DiscardPolicy:丢弃任务,但是不抛出异常。
- DiscardOldestPolicy:丢弃队列最前面的任务,然后重新提交被拒绝的任务
- CallerRunsPolicy:由调用线程(提交任务的线程)处理该任务
1.2.application.properties配置文件
universe.thread.pool.executor.rejectedExecutionHandler=java.util.concurrent.ThreadPoolExecutor.AbortPolicy
1.3.PoolExecutorConfig类的rejectedExecutionHandler字段
/**
* 当任务无法被执行时(超过线程最大容量 maximum 并且 workQueue 已经被排满了)的处理策略,
*/
@Value("${universe.thread.pool.executor.rejectedExecutionHandler}")
private String rejectedExecutionHandler;
1.4.通过反射获取rejectedExecutionHandlerClass 的类模板
什么是反射
Java的反射(reflection)机制是指在程序的运行状态中,可以构造任意一个类的对象,可以了解任意一个对象所属的类,可以了解任意一个类的成员变量和方法,可以调用任意一个对象的属性和方法。
通过反射反射获取对象的四种方式
- 通过类名.class获取
- 通过Class.forName名获取
- 通过对象.getclass获取
- 通过类加载器获取
这里我们用Class.forName(全类名)获取rejectedExecutionHandlerClass 的类模板
Class<?> rejectedExecutionHandlerClass = Class.forName(threadPoolExecutorConfig.getRejectedExecutionHandler());
1.5.我们将该字段传入Class.forName方法的时候,抛出了ClassNotFoundException异常。
三.问题排查
通过上网查阅资料我们可知内部类的全类名并非是包名.类.内部类而是包名.类$内部类。
四.解决问题
我们将配置文件的值由java.util.concurrent.ThreadPoolExecutor.AbortPolicy改为java.util.concurrent.ThreadPoolExecutor&AbortPolicy。如下所示
universe.thread.pool.executor.rejectedExecutionHandler=java.util.concurrent.ThreadPoolExecutor$AbortPolicy
五.测试
我们改好后重新启动服务,发现可以可以获取到了,测试通过。如下所示