递归实例化导致的栈溢出问题【简直蠢得出奇】

2023-12-02 21:04

本文主要是介绍递归实例化导致的栈溢出问题【简直蠢得出奇】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

问题描述

今天在练习数据库增删改查,体验三层架构思想时,随便写了点DAO层代码,但服务器运行时竟然爆出了栈溢出的问题,说实话,空指针问题我还能放着耐心去代码里找找问题,但这个栈溢出,我之前就没有一次解决过,要么改算法,要么直接重写出问题关联的那几段。至于网上说的改JVM的栈内存,这个我还不会。

先说说我遇到的问题吧,请看截图:
在这里插入图片描述

类型 异常报告消息 Servlet执行抛出一个异常描述 服务器遇到一个意外的情况,阻止它完成请求。例外情况javax.servlet.ServletException: Servlet执行抛出一个异常org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:51)
根本原因。java.lang.StackOverflowErrorjava.security.AccessController.doPrivileged(Native Method)org.apache.commons.logging.LogFactory.getContextClassLoaderInternal(LogFactory.java:808)org.apache.commons.logging.LogFactory.getFactory(LogFactory.java:419)org.apache.commons.logging.LogFactory.getLog(LogFactory.java:655)org.springframework.jdbc.support.JdbcAccessor.<init>(JdbcAccessor.java:43)org.springframework.jdbc.core.JdbcTemplate.<init>(JdbcTemplate.java:164)com.crud.crud_demo.dao.impl.EmployeeImpl.<init>(EmployeeImpl.java:16)com.crud.crud_demo.dao.impl.EmployeeImpl.<init>(EmployeeImpl.java:17)

我调用DAO层的EmployeeImpl来完成SQL语句来接收数据库数据,并以List集合返回给上层,最后返回到JSP页面层来显示。这里是我的EmployeeImpl代码:

package com.crud.crud_demo.dao.impl;import com.crud.crud_demo.dao.EmployeeDao;
import com.crud.crud_demo.domain.Employee;
import com.crud.crud_demo.util.JDBCUtils;
import org.springframework.jdbc.core.BeanPropertyRowMapper;
import org.springframework.jdbc.core.JdbcTemplate;import java.util.List;/*** 使用SQL语句的地方*/
public class EmployeeImpl implements EmployeeDao {//创建JdbcTemplate 对象并用静态工具类获取数据源private JdbcTemplate jdbcTemplate=new JdbcTemplate(JDBCUtils.getDataSource());private EmployeeDao employeeDao=new EmployeeImpl();@Overridepublic List<Employee> findAll() {String sql="select * from employee";List<Employee> list= jdbcTemplate.query(sql, new BeanPropertyRowMapper<Employee>(Employee.class));return list;}
}

发生原因

首先我们可以再看看看第一张服务器报错的界面
在这里插入图片描述
后面的报错地方全是一样的,说明这不是代码量较多导致变量过多的问题,而是某处存在创建变量的递归或者循环一直在运行。

相信聪明的小伙伴已经找到问题了,没错就是这一句:

private EmployeeDao employeeDao=new EmployeeImpl();

我也不知道这句我是什么时候写上去的,就很神奇。

EmployeeImpl类的实例化过程中,我通过private EmployeeDao employeeDao=new EmployeeImpl();

这一行代码,又创建了一个新的EmployeeImpl实例

而这个实例中的private EmployeeDao employeeDao=new EmployeeImpl();又会创建一个新的EmployeeImpl实例。

就这样无限递归下去,因为每次方法调用都会在调用栈上分配一定的空间,而无限递归会导致调用栈不断增长,直到耗尽可用的栈空间。

活脱脱的一个俄罗斯套娃
在这里插入图片描述

能写出这种💩代码试问这个世界上除了我还有谁能做到💩💩💩💩💩💩💩💩💩💩💩💩💩💩
在这里插入图片描述
不过借着这个机会了解了一下递归实例化的模式。也算是有点收获。

解决办法

删掉那个套娃语句就行

顺便了解一下什么是递归实例化

递归实例化是一种编程模式,它在某些情况下可能有用,但也存在一些潜在的缺点

优点:

简洁性:通过递归实例化,您可以使用较少的代码实现复杂的功能。相对于使用循环来处理嵌套结构,递归实例化的代码通常更简洁、易于理解和维护。

可读性:递归实例化可以使代码更加可读和自解释。它可以更直观地表示问题的解决方案,特别是对于涉及嵌套结构的问题。通过递归实例化,您可以将问题分解为更小的子问题,每个子问题都可以用相同的方式解决,从而使代码更具可读性。

灵活性:递归实例化可以应对未知深度的数据结构,因为它不需要提前知道要处理的嵌套层级。这使得递归实例化在处理树形结构、图形结构或其他具有递归性质的问题上非常有用。

缺点:

空间复杂度:递归实例化在函数调用过程中需要使用额外的内存来保存中间结果和函数调用栈。如果递归深度很大,这可能导致栈溢出或消耗大量内存。因此,在使用递归实例化时需要注意控制递归深度,以避免空间复杂度过高的问题。

时间复杂度:由于递归实例化需要进行多次函数调用和返回操作,因此可能会导致较高的时间复杂度。特别是在处理大量数据时,递归实例化可能会导致性能下降。因此,在使用递归实例化时需要注意优化算法以降低时间复杂度。

拜了,周末快乐

在这里插入图片描述

这篇关于递归实例化导致的栈溢出问题【简直蠢得出奇】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

线上Java OOM问题定位与解决方案超详细解析

《线上JavaOOM问题定位与解决方案超详细解析》OOM是JVM抛出的错误,表示内存分配失败,:本文主要介绍线上JavaOOM问题定位与解决方案的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一、OOM问题核心认知1.1 OOM定义与技术定位1.2 OOM常见类型及技术特征二、OOM问题定位工具

SpringBoot+RustFS 实现文件切片极速上传的实例代码

《SpringBoot+RustFS实现文件切片极速上传的实例代码》本文介绍利用SpringBoot和RustFS构建高性能文件切片上传系统,实现大文件秒传、断点续传和分片上传等功能,具有一定的参考... 目录一、为什么选择 RustFS + SpringBoot?二、环境准备与部署2.1 安装 RustF

javacv依赖太大导致jar包也大的解决办法

《javacv依赖太大导致jar包也大的解决办法》随着项目的复杂度和依赖关系的增加,打包后的JAR包可能会变得很大,:本文主要介绍javacv依赖太大导致jar包也大的解决办法,文中通过代码介绍的... 目录前言1.检查依赖2.更改依赖3.检查副依赖总结 前言最近在写项目时,用到了Javacv里的获取视频

Vue3绑定props默认值问题

《Vue3绑定props默认值问题》使用Vue3的defineProps配合TypeScript的interface定义props类型,并通过withDefaults设置默认值,使组件能安全访问传入的... 目录前言步骤步骤1:使用 defineProps 定义 Props步骤2:设置默认值总结前言使用T

深度解析Python中递归下降解析器的原理与实现

《深度解析Python中递归下降解析器的原理与实现》在编译器设计、配置文件处理和数据转换领域,递归下降解析器是最常用且最直观的解析技术,本文将详细介绍递归下降解析器的原理与实现,感兴趣的小伙伴可以跟随... 目录引言:解析器的核心价值一、递归下降解析器基础1.1 核心概念解析1.2 基本架构二、简单算术表达

Web服务器-Nginx-高并发问题

《Web服务器-Nginx-高并发问题》Nginx通过事件驱动、I/O多路复用和异步非阻塞技术高效处理高并发,结合动静分离和限流策略,提升性能与稳定性... 目录前言一、架构1. 原生多进程架构2. 事件驱动模型3. IO多路复用4. 异步非阻塞 I/O5. Nginx高并发配置实战二、动静分离1. 职责2

解决升级JDK报错:module java.base does not“opens java.lang.reflect“to unnamed module问题

《解决升级JDK报错:modulejava.basedoesnot“opensjava.lang.reflect“tounnamedmodule问题》SpringBoot启动错误源于Jav... 目录问题描述原因分析解决方案总结问题描述启动sprintboot时报以下错误原因分析编程异js常是由Ja

MySQL 表空却 ibd 文件过大的问题及解决方法

《MySQL表空却ibd文件过大的问题及解决方法》本文给大家介绍MySQL表空却ibd文件过大的问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录一、问题背景:表空却 “吃满” 磁盘的怪事二、问题复现:一步步编程还原异常场景1. 准备测试源表与数据

解决Nginx启动报错Job for nginx.service failed because the control process exited with error code问题

《解决Nginx启动报错Jobfornginx.servicefailedbecausethecontrolprocessexitedwitherrorcode问题》Nginx启... 目录一、报错如下二、解决原因三、解决方式总结一、报错如下Job for nginx.service failed bec

SysMain服务可以关吗? 解决SysMain服务导致的高CPU使用率问题

《SysMain服务可以关吗?解决SysMain服务导致的高CPU使用率问题》SysMain服务是超级预读取,该服务会记录您打开应用程序的模式,并预先将它们加载到内存中以节省时间,但它可能占用大量... 在使用电脑的过程中,CPU使用率居高不下是许多用户都遇到过的问题,其中名为SysMain的服务往往是罪魁