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

问题描述

今天在练习数据库增删改查,体验三层架构思想时,随便写了点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.mzph.cn/news/191072.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Vue学习笔记-Vuex基本使用

基本使用 初始化数据、配置actions、mutations&#xff0c;操作文件/store/index.js //index.js文件用于创建Vuex中最为核心的store对象 import Vue from vue import Vuex from vuex Vue.use(Vuex) //actions对象用于响应组件中的动作,专门负责业务逻辑 const actions {//函数…

Jetson Nano部署YOLOv5与Tensorrtx加速

一、烧录镜像 1、Jetson Nano烧写系统镜像 Jetson Nano是一款形状、外接口类似于树莓派的嵌入式主板&#xff0c;搭载了四核Cortex-A57处理器&#xff0c;GPU则是拥有128个NVIDIA CUDA核心的NVIDIA Maxwell架构显卡&#xff0c;内存为4GB的LPDDR4&#xff0c;存储则为16GB eM…

Postgresql和mysql的区别探究

PostgreSQL和MySQL是两个流行的关系型数据库管理系统&#xff08;RDBMS&#xff09;&#xff0c;具有各自的特点和优点。虽然两种数据库都可以处理大量数据&#xff0c;但在不同情况下&#xff0c;会有一些区别。下面将着重介绍PostgreSQL和MySQL的区别&#xff0c;并附带案例说…

入侵redis之准备---VMware上面安装部署centos7镜像系统【详细含云盘镜像】

入侵redis之准备—VMware上面安装部署centos7镜像系统【详细含云盘镜像 其他文章&#xff1a; 入侵redis并实现反弹shell控制【实战一】 学习大概步骤如下 第一步:先学习怎么安装部署kail系统服务器 入侵redis之准备—VMware安装部署kail镜像服务器【详细包含云盘镜像】 第二…

linux审计功能及规则 (audit.rule)

linux审计功能&#xff08;audit log&#xff09;是什么 audit是Linux自带的一套审计功能&#xff0c;可以监控我们日常的一些操作&#xff0c;然后将这些操作记录在audit.log中&#xff0c;方便我们查看日志。 审计规则是什么 在 /etc/audit/rules.d/audit.rules文件中&…

计网Lesson6 - IP 地址分类管理

文章目录 1. I P IP IP 地址定义2. I P v 4 IPv4 IPv4 的表示方法2.1 I P v 4 IPv4 IPv4 的分类编址法2.2 I P v 4 IPv4 IPv4 的划分子网法2.2.1 如何划分子网2.2.2 如何确定子网的借位数2.2.3 总结2.2.4 题目练习 2.3 I P v 4 IPv4 IPv4 的无分类编址法 1. I P IP IP 地…

04.里氏替换原则(Liskov Substitution Principle)

暴论&#xff1a;一般的&#xff0c;如果一个富二代不想着证明自己&#xff0c;那么他一辈子都会衣食无忧。 一言 里氏替换原则想告诉我们在继承过程中会遇到什么问题&#xff0c;以及继承有哪些注意事项。 概述 这是流传较广的一个段子&#xff1a; “一个坐拥万贯家财的富二…

亚马逊云科技Aurora MySQL在复制性能提升上的不断优化和尝试

前言 Amazon Aurora是亚马逊云科技自研的云原生关系数据库&#xff0c;它在提供和开源数据库MySQL、PostgreSQL的完好兼容性同时&#xff0c;也能够提供和商业数据库媲美的性能和可用性。 Aurora的性能提升不仅包含应用读写吞吐量的提升&#xff0c;也包含复制延迟的降低。一个…

代码随想录算法训练营第38天| 509. 斐波那契数 70. 爬楼梯 746. 使用最小花费爬楼梯

JAVA代码编写 动态规划&#xff08;Dynamic Programming&#xff09; 一个问题可以划分为多个子问题&#xff0c;且子问题之间有关联&#xff0c;就可以使用动态规划。 动态规划问题步骤&#xff1a; 确定dp数组&#xff08;dp table&#xff09;以及下标的含义确定递推公式…

栈顺序存储的实现(详解)

栈是一种数据结构&#xff0c;它具有后进先出&#xff08;LIFO&#xff09;的特性。栈可以用来存储一组元素&#xff0c;并且只能在栈顶进行插入和删除操作。栈的基本概念包括&#xff1a; 1. 入栈&#xff08;push&#xff09;&#xff1a;将元素添加到栈顶的操作。 2. 出栈&…

图片消除笔哪些软件有?这三款智能消除软件值得收藏

图片消除笔哪些软件有&#xff1f;日常在社交媒体上分享照片或制作海报&#xff0c;广告时&#xff0c;经常会遇到需要删除图片中多余的元素&#xff0c;比如水印、日期、人物等&#xff0c;以便更好的去将这些图片进行二次创作&#xff0c;那么有哪些软件有图片消除笔可以选择…

MySQL5.7安装与配置:自动化一键安装配置

介绍 本文介绍了一个自动化安装MySQL的Shell脚本。该脚本可以帮助用户快速安装MySQL&#xff0c;并自动进行配置和初始化。通过使用该脚本&#xff0c;用户无需手动执行繁琐的安装步骤&#xff0c;大大简化了MySQL的安装过程。 使用shell自动化安装教程 1. 复制脚本 首先&a…

webGIS使用JS,高德API完成简单的智慧校园项目基础

代码实现 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name"viewport" content"widthdevice-width, i…

基于jsonrpc4j实现JSON-RPC over TCP

1.JSON-RPC说明 JSON-RPC是一个无状态且轻量级的远程过程调用(RPC)协议。 它主要定义了一些数据结构及其相关的处理规则。 它运行时可以基于tcp(socket),http等不同的消息传输方式&#xff0c; 即它不关心底层传输方式的细节。 它使用JSON&#xff08;RFC 4627&#xff09;作为…

Java研学-配置文件

一 配置文件 1 作用–解决硬编码的问题 在实际开发中,有时将变量的值直接定义在.java源文件中;如果维护人员想要修改数据,无法完成(因为没有修改权限),这种操作称之为硬编码 2 执行原理: 将经常需要改变的数据定义在指定类型的文件中,通过java代码对指定的类型的文件进行操作…

鸿蒙学习之TypeScript 语法理解笔记

1、变量及数据类型 // string&#xff1a;字符串&#xff0c;单引号或双引号 let msg : string hello wprld console.log(msg:msg)// number&#xff1a;数值、整数、浮点let num :number 21console.log(num:num)//boolean&#xff1a;布尔let finished: boolean truecons…

读书笔记:《Effective Modern C++(C++14)》

Effective Modern C&#xff08;C14&#xff09; GitHub - CnTransGroup/EffectiveModernCppChinese: 《Effective Modern C》- 完成翻译 Deducing Types 模版类型推导&#xff1a; 引用&#xff0c;const&#xff0c;volatile被忽略数组名和函数名退化为指针通用引用&#…

java学习part30callabel和线程池方式

140-多线程-线程的创建方式3、4&#xff1a;实现Callable与线程池_哔哩哔哩_bilibili 1.Callable 实现类 使用方式 返回值 2.线程池

DPDK-Hello-World示例应用程序

文章目录 前言环境安装编写hello-world程序 前言 目标: 在linux上安装DPDK的程序编写环境&#xff0c;编写和运行DPDK的hello world程序。 声明&#xff1a;我不清楚DPDK具体是个啥。DPDK的目的大概是&#xff1a;原先的网络数据需要从内核层拷贝到用户层&#xff0c;在IO越来…

检测判断IP合法性API接口

检测判断IP合法性API接口 一、检测判断IP合法性API接口二、使用步骤1、接口2、请求参数3、请求参数示例4、接口 返回示例 三、 如何获取appKey和uid1、申请appKey:2、获取appKey和uid 四、重要说明 一、检测判断IP合法性API接口 一款免费的帮助你检测判断IP合法性API接口 二、…