StackOverFlow异常记录

2024-04-18 13:08
文章标签 异常 记录 stackoverflow

本文主要是介绍StackOverFlow异常记录,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在写cglib动态代理实现aop的代码的时候,发现运行测试程序报出了stackoverflow异常,话不多说直接上代码:

CGLibProxy实现如下:

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;import java.lang.reflect.Method;/*** CGLibProxy动态代理类的实例* 被代理类是否实现接口对cglib实现没有影响**/
public class CGLibProxy implements MethodInterceptor {/*  @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy)throws Throwable {return null;}*///根据目标对象生成一个子类作为他的代理类public Object createProxyObject(Object obj) {Enhancer enhancer = new Enhancer();enhancer.setSuperclass(obj.getClass());//设置父类为被代理类enhancer.setCallback(this);Object proxyObj = enhancer.create();//create 方法里用这个Superclass生成了子类return proxyObj;// 返回代理对象// 返回代理对象,返回的对象其实就是一个封装了“实现类”的代理类,是实现类的实例。}public Object intercept(Object proxy, Method method, Object[] args,MethodProxy methodProxy) throws Throwable {Object obj = null;if ("addUser".equals(method.getName())) {// 过滤方法checkPopedom();// 检查权限}obj = methodProxy.invoke(proxy, args);return obj;}private void checkPopedom() {System.out.println(".:检查权限  checkPopedom()!");}
}

测试类如下:

public class Test {public static void main(String[] args) {Test t = new Test();t.test();}private void test() {UserManager userManager = (UserManager) new CGLibProxy().createProxyObject(new UserManagerImpl());System.out.println("-----------CGLibProxy-------------");userManager.addUser("tom", "root");}}

异常信息如下:

在反复打印检查权限之后,报出了stackoverflow异常。

问题到底出在哪儿呢?

仔细查看以及debug自己的代码之后,发现问题出在下面这一行:

 obj = methodProxy.invoke(proxy, args);

通过反射去掉用传入对象的增强方法,而这里传入的对象是proxy类,而proxy类自己又回调用这个增强方法,因此无限的调用增强方法,导致了stackoverflow的出现。

修改方式:

在实现动态代理的时候,在增强方法里实现了增强逻辑之后,需要调用的是被代理类本身需要执行的方法,因此这里应该传入的不是proxy对象,而是在createPrxoxy方法中指定需要代理的类的对象。修改代码如下:

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;import java.lang.reflect.Method;/*** CGLibProxy动态代理类的实例* 被代理类是否实现接口对cglib实现没有影响**/
public class CGLibProxy implements MethodInterceptor {/*  @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy)throws Throwable {return null;}*///根据目标对象生成一个子类作为他的代理类private Object target;public Object createProxyObject(Object obj) {Enhancer enhancer = new Enhancer();target = obj;enhancer.setSuperclass(obj.getClass());//设置父类为被代理类enhancer.setCallback(this);Object proxyObj = enhancer.create();//create 方法里用这个Superclass生成了子类return proxyObj;// 返回代理对象// 返回代理对象,返回的对象其实就是一个封装了“实现类”的代理类,是实现类的实例。}public Object intercept(Object proxy, Method method, Object[] args,MethodProxy methodProxy) throws Throwable {Object obj = null;if ("addUser".equals(method.getName())) {// 过滤方法checkPopedom();// 检查权限}obj = methodProxy.invoke(target, args);return obj;}private void checkPopedom() {System.out.println(".:检查权限  checkPopedom()!");}
}

再运行测试类就成功了。

 

 

 

 

 

 

 

 

这篇关于StackOverFlow异常记录的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/914868

相关文章

Java捕获ThreadPoolExecutor内部线程异常的四种方法

《Java捕获ThreadPoolExecutor内部线程异常的四种方法》这篇文章主要为大家详细介绍了Java捕获ThreadPoolExecutor内部线程异常的四种方法,文中的示例代码讲解详细,感... 目录方案 1方案 2方案 3方案 4结论方案 1使用 execute + try-catch 记录

国内环境搭建私有知识问答库踩坑记录(ollama+deepseek+ragflow)

《国内环境搭建私有知识问答库踩坑记录(ollama+deepseek+ragflow)》本文给大家利用deepseek模型搭建私有知识问答库的详细步骤和遇到的问题及解决办法,感兴趣的朋友一起看看吧... 目录1. 第1步大家在安装完ollama后,需要到系统环境变量中添加两个变量2. 第3步 “在cmd中

Spring Retry 实现乐观锁重试实践记录

《SpringRetry实现乐观锁重试实践记录》本文介绍了在秒杀商品SKU表中使用乐观锁和MybatisPlus配置乐观锁的方法,并分析了测试环境和生产环境的隔离级别对乐观锁的影响,通过简单验证,... 目录一、场景分析 二、简单验证 2.1、可重复读 2.2、读已提交 三、最佳实践 3.1、配置重试模板

在 Spring Boot 中使用异步线程时的 HttpServletRequest 复用问题记录

《在SpringBoot中使用异步线程时的HttpServletRequest复用问题记录》文章讨论了在SpringBoot中使用异步线程时,由于HttpServletRequest复用导致... 目录一、问题描述:异步线程操作导致请求复用时 Cookie 解析失败1. 场景背景2. 问题根源二、问题详细分

解决java.lang.NullPointerException问题(空指针异常)

《解决java.lang.NullPointerException问题(空指针异常)》本文详细介绍了Java中的NullPointerException异常及其常见原因,包括对象引用为null、数组元... 目录Java.lang.NullPointerException(空指针异常)NullPointer

关于Spring @Bean 相同加载顺序不同结果不同的问题记录

《关于Spring@Bean相同加载顺序不同结果不同的问题记录》本文主要探讨了在Spring5.1.3.RELEASE版本下,当有两个全注解类定义相同类型的Bean时,由于加载顺序不同,最终生成的... 目录问题说明测试输出1测试输出2@Bean注解的BeanDefiChina编程nition加入时机总结问题说明

Spring Boot统一异常拦截实践指南(最新推荐)

《SpringBoot统一异常拦截实践指南(最新推荐)》本文介绍了SpringBoot中统一异常处理的重要性及实现方案,包括使用`@ControllerAdvice`和`@ExceptionHand... 目录Spring Boot统一异常拦截实践指南一、为什么需要统一异常处理二、核心实现方案1. 基础组件

将sqlserver数据迁移到mysql的详细步骤记录

《将sqlserver数据迁移到mysql的详细步骤记录》:本文主要介绍将SQLServer数据迁移到MySQL的步骤,包括导出数据、转换数据格式和导入数据,通过示例和工具说明,帮助大家顺利完成... 目录前言一、导出SQL Server 数据二、转换数据格式为mysql兼容格式三、导入数据到MySQL数据

关于rpc长连接与短连接的思考记录

《关于rpc长连接与短连接的思考记录》文章总结了RPC项目中长连接和短连接的处理方式,包括RPC和HTTP的长连接与短连接的区别、TCP的保活机制、客户端与服务器的连接模式及其利弊分析,文章强调了在实... 目录rpc项目中的长连接与短连接的思考什么是rpc项目中的长连接和短连接与tcp和http的长连接短

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI