初始JavaEE篇 —— Spring Web MVC入门(下)

news/2025/2/9 9:48:02/

 找往期文章包括但不限于本期文章中不懂的知识点:

个人主页:我要学编程程(ಥ_ಥ)-CSDN博客

所属专栏:JavaEE

初始JavaEE篇 —— Spring Web MVC入门(上)

在上篇文章中,我们学习了一些注解的使用、Postman模拟前端传递数据的使用以及如何在后端接受并处理前端的数据。接下来我们就需要学习响应的返回,将前端的数据处理完成之后,就需要返回数据给前端。

目录

返回响应数据

返回静态页面

返回普通数据 

返回HTML代码片段

返回JSON数据 

设置状态码 

设置Header


返回响应数据

Http请求的响应结果可以是数据,也可以是静态页面,也可以针对响应设置状态码、Header信息等。

返回静态页面

返回静态页面需要学习一个新的注解:@Controller,这个注解与@RestController 有所不同。@RestController 表示该控制器的所有方法都会默认返回 JSON 或其他对象,而不是解析和返回视图页面。而@Controller 就是按照页面来解析的。

前端代码:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>Index</title>
</head>
<body><h1>Hello World</h1>
</body>
</html>

后端代码:

@RequestMapping("/Response")
@Controller
public class ResponseController {// 返回静态页面@RequestMapping("/index")public String index() {return "/index.html";}
}

这里的 "/" 代指的是 src/main/resource/static 目录下。可能会存在多个目录,那时就需要将完整路径显示出来。

访问的结果:

但如果我们使用@RestController注解的话,就只会返回一个字符串。这就是两者在返回数据时,会通过注解的不同,从而返回不同的资源。@Controller注解只会返回视图,而@ResponseBody注解会返回JSON数据或对象。 

注意:

1、前端代码在Spring项目中,是存放在 src/main/resource/static 目录下的。

2、@RestController = @Controller + @ResponseBody,也就是说@RestCont注解是两者的结合体,我们也可以通过@RestController的源码来观察:

注解应用类是一个枚举类,源代码如下(了解即可):

public enum ElementType {/** Class, interface (including annotation interface), enum, or record* declaration */TYPE,/** Field declaration (includes enum constants) */FIELD,/** Method declaration */METHOD,/** Formal parameter declaration */PARAMETER,/** Constructor declaration */CONSTRUCTOR,/** Local variable declaration */LOCAL_VARIABLE,/** Annotation interface declaration (Formerly known as an annotation type.) */ANNOTATION_TYPE,/** Package declaration */PACKAGE,/*** Type parameter declaration** @since 1.8*/TYPE_PARAMETER,/*** Use of a type** @since 1.8*/TYPE_USE,/*** Module declaration.** @since 9*/MODULE,/*** Record component** @jls 8.10.3 Record Members* @jls 9.7.4 Where Annotations May Appear** @since 16*/RECORD_COMPONENT;
}

注解的生命周期有三个阶段:源码、字节码(编译)、运行。

public enum RetentionPolicy {/*** Annotations are to be discarded by the compiler.*/SOURCE, // 只存在于源码阶段/*** Annotations are to be recorded in the class file by the compiler* but need not be retained by the VM at run time.  This is the default* behavior.*/CLASS, // 既存在于源码,也存在于字节码(编译)阶段/*** Annotations are to be recorded in the class file by the compiler and* retained by the VM at run time, so they may be read reflectively.** @see java.lang.reflect.AnnotatedElement*/RUNTIME // 既存在于源码阶段,也存在于字节码阶段,还存在与运行时
}

返回普通数据 

我们前面学习请求时,返回的响应都是字符串数据,使用的注解是@RestController,这里就包含了@ResponseBody 注解,因而可以返回数据。

如果我们现在在之前的代码的基础上,加上一个返回字符串数据的代码会成功返回吗?

    @RequestMapping("/login")public String login(String userName, String password) {return "服务器异常,请稍后重试~";}

我们会发现,根本访问不了, 直接告诉我们资源不存在。因为这里的注解是@Controller说明是需要返回的是视图,那么Spring就会去扫描视图,但是最终却不存在这个视图,即404。

解决方法是在原本的方法前面加上@ResponseBody注解,这样最终就是@RestController注解了。

    @ResponseBody@RequestMapping("/login")public String login(String userName, String password) {return "服务器异常,请稍后重试~";}

@ResponseBody既是类注解,又是方法注解。如果作用在类上,表示该类的所有方法,返回的都是数据,如果作用在方法上,表示该方法返回的是数据。

返回HTML代码片段

后端的代码中,如果带有HTML代码片段就会被浏览器解析出来,但我们不想要浏览器解析,该怎么做呢?

    @ResponseBody@RequestMapping("/returnHtml")public String returnHtml() {return "<h1>Hello World</h1>";}

 上述代码在浏览器中访问就会被解析成下面这样:

之所以将HTML标签解析成具体可视化页面,是因为响应数据的返回格式是:text/html,所以浏览器就会解析HTML标签。那如果我们将返回数据返回设置成其他的数据是否就可以让浏览器不再按照HTML标签来解析呢?我们可以使用fiddler对Http请求进行抓包处理。下图是修改之前的结果:

修改之后的结果:

代码:

@ResponseBody
// produces-设置方法返回的数据类型或内容类型
@RequestMapping(value = "/returnHtml",produces = "text/plain")
public String returnHtml(HttpServletResponse response) {return "<h1>Hello World</h1>";
}

浏览器的结果: 

我们想要设置成什么类型,可以直接在搜索引擎上去找或者问AI即可。 

返回JSON数据 

我们前面已经学习了,后端在传输对象时,都是Spring框架使用Jackson将对象序列化为JSON数据从而进行传输。

    @ResponseBody@RequestMapping("/returnJSON")public UserInfo returnJSON() {return new UserInfo();}

设置状态码 

SpringMVC会根据我们方法的返回结果自动设置响应状态码,程序员也可以手动指定状态码。通过SpringMVC的内置对象HttpServletResponse提供的方法来进行设置。

    @ResponseBody@RequestMapping("/setStatus")public String setStatus(HttpServletResponse response) {response.setStatus(500);return "服务器一切正常,但是就是返回500!嘿嘿~";}

虽然状态码被设置成了非正常的情况,但是响应依旧是正常的,即状态码不会影响正常响应结果。 

设置Header

Http响应报头也会向客户端传递一些附加信息,比如:服务程序的名称,请求的资源,已移动到新地址等,如:Content-Type等信息。这些信息通过 @RequestMapping 注解的属性来实现。即设置Header信息都是通过@RequestMapping注解的属性来实现的。
先来看@RequestMapping的源码: 

@Target({ElementType.TYPE, ElementType.METHOD}) // 类注解、方法注解
@Retention(RetentionPolicy.RUNTIME) // 源码、字节码、运行阶段皆存在
@Documented
@Mapping
@Reflective(ControllerMappingReflectiveProcessor.class)
public @interface RequestMapping {// 用来指定映射的名称(例如,"/r1"等)String name() default "";// value 和 path 都是指定映射的URL@AliasFor("path")String[] value() default {};@AliasFor("value")String[] path() default {};// 指定请求的类型RequestMethod[] method() default {};// 指定请求中必须包含某些参数值(只有包含时,才处理)String[] params() default {};// 用来指定请求中的头部信息(只有包含指定信息时,才处理)String[] headers() default {};// 用来指定请求的 Content-TypeString[] consumes() default {};// 用来指定方法返回的 Content-TypeString[] produces() default {};}

 注意:name 和 value 区别:

@RequestMapping(name = "myMapping", value = "/api/user")
public String getUser() {return "user";
}

上述代码中,name用作映射的标识符,通常用于生成文档、日志或调试。value是实际请求的URL。在开发中,我们一般不会去使用name属性。

通过设置 produces 属性的值,来设置相应的报头Content-Type:

    @ResponseBody@RequestMapping(value = "/setContentType",produces = "application/json")public String setContentType() {return "{\"name\": \"zhangsan\"}";}

上述代码是将响应报头的Content-Type设置成JSON数据,这样最终返回的响应数据就会按照与之对应的数据来解析。

注意:在Java中," 是属于 特殊字符,所以需要使用反斜杠(\)进行转义。正常的JSON数据是下面这样的:

{"name": "zhangsan"}

经过转义之后的结果就是下面这样(在 " 前面加上 \ 即可):

{\"name\": \"zhangsan\"}

除了可以设置原本的属性之外,还可以手动添加新的属性到Header中:

    @ResponseBody@RequestMapping("setHeader")public String setHeader(HttpServletResponse response) {response.setHeader("name", "zhangsan");return "header设置成功";}

好啦!本期 初始JavaEE篇 —— Spring Web MVC入门(下)的学习之旅 就到此结束啦!我们下一期再一起学习吧!


http://www.ppmy.cn/news/1570550.html

相关文章

Android原生开发问题汇总

Fragment顶部出现一个白条怎么办&#xff1f;父类布局搞事情。 layer-list被拉伸问题 Android之 ImageView android:src和tools:src的区别是什么? Android运行时权限的总结&#xff0c;以及EasyPermissions框架的使用 Android Studio添加EasyPemissions Android中module怎…

python基础入门:3.2字典(Dict)与集合(Set)

Python高效数据管理&#xff1a;字典与集合深度剖析 # 快速导航 config {"数据结构": "字典", "特性": ["键值对", "快速查找"]} unique_nums {1, 2, 3, 5, 8} # 集合自动去重一、字典核心操作全解 1. 键值对基础操作 …

MySQL数据库(七)SQL 优化

一 插入数据 采用方法 1 批量插入 2 手动提交事务 3 主键顺序插入 4* 使用load插入指令数据 二 主键优化 1 数据组织方式 在InnoDB存储引擎中&#xff0c;表中的数据都是根据主键顺序组织存放的&#xff0c;这种存储方式的表称为索引组织表 2 页分裂 页可以为空也可…

光伏-报告显示,假期内,硅料端签单顺序发货相对稳定。若3月份下游存提产,则不排除硅料价格有上调预期。

据TrendForce集邦咨询报告显示&#xff0c;假期内&#xff0c;硅料端按照前期签单顺序发货&#xff0c;相对稳定。若3月份下游存提产&#xff0c;则不排除硅料价格有上调预期。 002306中科云网 旅游 | 公司为提供复合菜系特色餐饮的连锁企业&#xff0c;形成了以粤菜&#xff…

Kotlin 的作用域函数:apply、let、also、run,以及 with、takeIf 和 takeUnless。函数的用法和使用场景

Kotlin中提供几个常用的作用域函数,如果能将它们熟练运用,那将会有效的提高开发效率,降低开发时间,同时保证程序的稳定。那分别是哪几个作用域函数? Kotlin 的作用域函数:apply、let、also、run,以及 with、takeIf 和 takeUnless。这些函数是编写简洁、富有表现力的 Ko…

Win10 部署llama Factory 推荐教程和遇到的问题

教程 【大模型微调】使用Llama Factory实现中文llama3微调_哔哩哔哩_bilibili 大模型微调&#xff01;手把手带你用LLaMA-Factory工具微调Qwen大模型&#xff01;有手就行&#xff0c;零代码微调任意大语言模型_哔哩哔哩_bilibili 遇到问题解决办法 pytorch gpu国内镜像下载…

101.对称二叉树 python

对称二叉树 题目题目描述示例 1&#xff1a;示例 2&#xff1a;提示&#xff1a; 题解递归法步骤提交结果 迭代法步骤提交结果 题目 题目描述 给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 示例 1&#xff1a; 输入&#xff1a;root [1,2,2,3,4,4,3] 输出…

客户端脚本安全设置:如何保障您的Web应用免受攻击?

随着现代Web开发的不断发展&#xff0c;客户端脚本在增强用户互动和提升网站体验方面扮演着重要角色。然而&#xff0c;这些便利的背后也伴随着潜在的安全风险&#xff0c;恶意攻击者可能会利用这些漏洞进行攻击。因此&#xff0c;确保客户端脚本的安全至关重要。本文将带您了解…