SpringMVC Controller 层没有使用 @ResponseBody 注解引发的血案(api访问404)

本文主要是介绍SpringMVC Controller 层没有使用 @ResponseBody 注解引发的血案(api访问404),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

问题现象:

项目组的一个同事发现在请求该接口时候,总是报 404 错误,又找不到错误日志,一时之间不知道该如何去着手解决问题,我帮他排查问题的时候,发现该接口两次经过拦截器的 preHandle 方法,并且通过 debug 断点调试发现,两次请求的路径不一样,这一下我也不知道该如何解决该问题了,现象如下:

现象复现测试代码,非原代码(原代码不宜公开):

@Controller
@RequestMapping("/api")
public class TestController {@ApiOperation(httpMethod = "GET", value = "注销登录", notes = "注销登录")public String testController() {return "成功";}}

测试结果:

{"timestamp": "2024-04-19T11:34:02.499+00:00","status": 404,"error": "Not Found","message": "","path": "/api/test-controller"
}

测试过程,debugger 情况展示:

在这里插入图片描述
很明显两次请求的路径不一样,第一次请求的路径是我们正常的路径:”/api/test-controller“,第二次请求的路径正是:“/api/成功”,很明显问题就出在第二请求路径上,代码中不存在这个请求路径,肯定要报错:404,为什么会有第二次请求呢?

原因分析:

第二次请求肯定不是我们自己请求的,那必然是程序自己触发的,也就是 SpingMVC 自己发起的,还是老办法使用 debugg 调试,在进行 debugger 调试跟踪源码的时候,发现第一次请求把返回值封装成了 ModelAndView 对象,等于返回值成了一个待跳转的页面,至此就很好理解 404 错误了。

在这里插入图片描述

找到原因了就很好解决问题了,导致这个问题的根源就是没有使用 @ResponseBody,因为习惯了使用 @RestController 注解,想着控制层Controller 代码很简单,应该不会出这个问题,同事可能也是复制之前的代码,一是没有注意导致了这个问题的发生。

@RestController 相当于 @Controller + @ResponseBody 注解,源码如下:

package org.springframework.web.bind.annotation;import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import org.springframework.core.annotation.AliasFor;
import org.springframework.stereotype.Controller;@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Controller
@ResponseBody
public @interface RestController {@AliasFor(annotation = Controller.class)String value() default "";
}

欢迎提出建议及对错误的地方指出纠正。

这篇关于SpringMVC Controller 层没有使用 @ResponseBody 注解引发的血案(api访问404)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

Linux使用dd命令来复制和转换数据的操作方法

《Linux使用dd命令来复制和转换数据的操作方法》Linux中的dd命令是一个功能强大的数据复制和转换实用程序,它以较低级别运行,通常用于创建可启动的USB驱动器、克隆磁盘和生成随机数据等任务,本文... 目录简介功能和能力语法常用选项示例用法基础用法创建可启动www.chinasem.cn的 USB 驱动

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

java脚本使用不同版本jdk的说明介绍

《java脚本使用不同版本jdk的说明介绍》本文介绍了在Java中执行JavaScript脚本的几种方式,包括使用ScriptEngine、Nashorn和GraalVM,ScriptEngine适用... 目录Java脚本使用不同版本jdk的说明1.使用ScriptEngine执行javascript2.

c# checked和unchecked关键字的使用

《c#checked和unchecked关键字的使用》C#中的checked关键字用于启用整数运算的溢出检查,可以捕获并抛出System.OverflowException异常,而unchecked... 目录在 C# 中,checked 关键字用于启用整数运算的溢出检查。默认情况下,C# 的整数运算不会自

在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码

《在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码》在MyBatis的XML映射文件中,trim元素用于动态添加SQL语句的一部分,处理前缀、后缀及多余的逗号或连接符,示... 在MyBATis的XML映射文件中,<trim>元素用于动态地添加SQL语句的一部分,例如SET或W

Mybatis官方生成器的使用方式

《Mybatis官方生成器的使用方式》本文详细介绍了MyBatisGenerator(MBG)的使用方法,通过实际代码示例展示了如何配置Maven插件来自动化生成MyBatis项目所需的实体类、Map... 目录1. MyBATis Generator 简介2. MyBatis Generator 的功能3