Spring基础——使用注解开发SpringMVC

news2024/12/27 5:15:08

目录

  • 配置SpringMVC的初始化信息
    • 配置ServletWebApplicationContext
    • 配置RootWebApplicationContext
    • 配置ServletContext
  • 创建Controller控制器
    • 配置Controller响应路径
    • 接收用户传递参数
      • 接收JSON数据
      • 接收简单类型
        • 对象封装参数
      • 接收数组类型
  • Restful

文章源码仓库:Spring Learn Repo

配置SpringMVC的初始化信息

  • 之前说过,SpringMVC中使用两个WebApplicationContext:ServletWebApplicationContext和RootWebApplicationContext对Web上下文环境进行分层管理。
    • ServletWebApplicationContext:用于管理Servlet相关的Bean,如控制器,视图解析器等,每个Servlet(通常是DispatcherServlet)都有自己的ServletWebApplicationContext;并且Scope与Servlet相对应
    • RootWebApplicationContext:是整个Web应用的上下文,用于管理应用程序级别的Bean,如Service,Repositories等,RootWebApplicationContext可以被多个Servlet共享,且Scope与整个Web应用相同。
      SpringMVC

配置ServletWebApplicationContext

  • 上文提到过每个DispatcherServlet都有一个ServletWebApplicationContext用于存储Servlet的相关Bean,因此我们需要扫描Controller注解标记的Bean
  • @EnableWebMvc:值得注意的是,这里还需要Spring MVC开启基于注解的基础功能以及一些默认配置。
    • 通过此注解SpringMVC会自动注册包括DefaultAnnotationHandlerMapping,AnnotationMethodHandlerAdapter等用于处理注解映射和处理方法的适配器。
    • 并且允许在Controller标记类的类和方法上使用@RequestMapping,@RequestParam等HTTP请求与响应方面的注解驱动如果你发现RequestParam没法使用的话可以查看是否开启了此注解
    • 而且此注解还提供了一些常见的用于处理请求参数的类型转换,如字符串转数字,json格式之间的转换等
@Configuration
@ComponentScan({"com.nobugnolife.controller"})
@EnableWebMvc
public class SpringMvcConfig {
}

配置RootWebApplicationContext

  • 上文说过RootWebApplicationContext里面是Web用于管理应用程序级别的Bean,如Service,Repositories等,而我们这里只是演示SpringMVC的功能,所以就不配置持久层了
@Configuration
@ComponentScan({"com.nobugnolife.service"})
public class SpringConfig {
}

配置ServletContext

  • Spring提供了WebApplicationInitializer接口可以让我们注册并初始化DispatcherServlet
public class MyWebApplicationInitializer implements WebApplicationInitializer {

    @Override
    public void onStartup(ServletContext servletContext) {

        // 加载ServletWebApplicationContext配置信息
        AnnotationConfigWebApplicationContext context = new AnnotationConfigWebApplicationContext();
        // 注册SpringMVC的配置类
        context.register(SpringMvcConfig.class);

        // 创建并注册DispatcherServlet
        DispatcherServlet servlet = new DispatcherServlet(context);
        ServletRegistration.Dynamic registration = servletContext.addServlet("app", servlet);
        registration.setLoadOnStartup(1);
        registration.addMapping("/");
    }
}
  • 官方文档里提供了一个简化的创建方案通过继承 AbstractAnnotationConfigDispatcherServletInitializer 并覆写特定的方法,用以简化配置开发
public class ServletConfig extends AbstractAnnotationConfigDispatcherServletInitializer {
    @Override
    protected Class<?>[] getRootConfigClasses() {
        return new Class[]{SpringConfig.class};
    }

    @Override
    protected Class<?>[] getServletConfigClasses() {
        return new Class[]{SpringMvcConfig.class};
    }

    @Override
    protected String[] getServletMappings() {
        return new String[]{"/"};
    }

    /*
    * 过滤器
    * */
}

创建Controller控制器

  • 在控制层Controller中我们需要运用到如下这些注解:
    • @Controller:Controller Bean注解,用于标记controller中的类
    • @RequestMapping:在之前的文章中说过SpringMVC对用户发出的请求会有HandlerMapping解析路由选择对应处理器进行处理,因此我们在创建处理器Controller的时候也需要提供处理器对应的URL路径,通过 @RequestMapping,如果不提供默认为当前上下文的根路径
      • @PostMapping:@RequestMapping(value=“”, method = ReuqestMethod.POST)简化写法
      • @PutMapping:@RequestMapping(value=“”, method = ReuqestMethod.PUT)简化写法
      • @DeleteMapping:@RequestMapping(value=“”, method = ReuqestMethod.DELETE)简化写法
      • @GetMapping:@RequestMapping(value=“”, method = ReuqestMethod.GET)简化写法
    • @RequestParam:通过路径传参,允许接受用户从URL路径中传入的参数,并写入到形参中如:https://www.baidu.com/search?context=csdn,那么这里传入了个参数名为context,内容为csdn的信息,并且在SpringMVC中会自动转化成形参类型
    • @RequestBody:通过请求体传参,允许接受用户将信息写入请求体中,可以避免直接从URL中暴露参数信息,但一般接收的时候会通过post方式进行接收
    • @PathVariable:URL模板变量,属于简化了URL路径传参的方式,允许用户通过URL路径解析的格式将参数动态传入,但需要指明传入参数的位置如:https://www.baidu.com/search/{context},这里用户传入参数只需要在search/csdn,SpringMVC会自动将{context}路径对应的csdn解析到context参数中
    • @ResponseBody:将返回数据直接作为响应参数进行返回,之前文章说过,在SpringMVC中,DispatcherServlet会将我们最终处理完的数据注入到视图模型中并返回给用户,因此正常来说Controller的返回数据会被当成URL路径解析,而如果我们不想返回视图,只想返回我们的处理数据的话则需要用到这个注解
    • @RestController:将@ResponseBody与@Controller进行合并,用于简化开发代码

配置Controller响应路径

  • 通过@RequestMapping配置URL的解析路径(从SpringMCV的ServletContext配置的根路径开始解析),@RequestMapping可以作用到类上(作用到类上代表当前Controller类中所有的方法都会通过此路径),也可以作用到方法上(方法上的路径会自动配置到类的路径后边)
  • 比如这里我们配置一个WebController,通过访问/index/hello资源路径让服务器返回跳转到index.jsp
@Controller
@RequestMapping("/index")
public class WebController {

    /*
    * 如果没有添加RequestBody则会把返回字符串当成路径访问服务器文件
    * */
    @RequestMapping("/hello")
    public String toIndex(){
        System.out.println("跳转到index.jsp");
        return "index.jsp";
    }
}

接收用户传递参数

接收JSON数据

  • JSON是一种轻量级的数据交换格式,基于JavaScript语法的子集,但如今是可独立于JavaScript使用,通常用于Web应用程序中的数据交换和存储,包括Restful中的数据传输,Ajax请求的数据格式,配置文件等,因为其简洁性和可读性,也被广泛用于日志,数据序列化等领域
    • 简洁性:JSON使用简洁的文本格式表示数据
    • 可读性:JSON数据格式有良好的可读性,易于被理解
    • 自我描述性:JSON支持多种数据类型,包括数字,字符串,数组,对象等,可以表示丰富的数据结构
    • 易于解析:JSON数据可以通过大多数编程语言的解析器轻松解析为内部数据结构
  • JSON的数据由键值对组成,其中键(Key)为字符串,值(Value)为表示的数据,一个完整的JSON对象使用{}表示,键值对之间使用:分割,JSON中数组使用[]表示,不同元素之间使用,分割,以下为JSON的简单示例:
{
  "name": "John",
  "age": 30,
  "isStudent": false,
  "friends": ["Alice", "Bob", "Charlie"],
  "address": {
    "city": "New York",
    "zipcode": "10001"
  }
}
  • 在SpingMVC中要想接收JSON数,官方文档给出Spring提供支持Jackson Json的依赖但是功能比较简单,不具备Jackson库提供的一些高级特性,同样SpringMVC也可以使用其他JSON转化库如:
    • Gson:Goolge提供的JSON转化,功能齐全,转化方便
    • FastJson:阿里提供的高性能JSON转化,转化非常快,但转化复杂类型的对象容易出错,可以用Gson将Bean转为Json保证数据正确,用FastJson将Json转Bean提高性能
    • Jackson Json:Jackson是一个成熟的,广泛使用的JSON处理库,其中提供了丰富的功能机制,也支持各种高级特性,在性能上处理大量JSON数据格式会比Gson更快,在支持性与功能性上又比FastJson更多
  • 这里我们导入Jackson库
<properties>
    <jackson.version>2.15.2</jackson.version>
</properties>

<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>${jackson.version}</version>
</dependency>
  • SpringMVC使用@RequestBody会自动将传入的Json数据转化成对象,并且会对有@ResponseBody的方法返回值自动转化成Json数据返回
@Data
public class Product {
    private String productName;
    private Integer productId;
    private Double value;
}
@PostMapping
public String addProduct(@RequestBody Product product) {
    System.out.println("请求访问商品添加接口,当前商品信息为:" + product);
    return "商品信息添加成功";
}
  • 测试数据
    test_json
    test_json_result

接收简单类型

  • 对于从URL传入的参数,SpringMVC能自动传入到对应的参数名中同时也可以使用@RequestParam对传入的参数名与形参名进行映射
@RequestMapping("/login")
@ResponseBody
public String userLogin(@RequestParam("username") String username, String password) {
    if (Objects.equals(password, "123")) {
        System.out.println(username + "登录成功");
    }else {
        System.out.println("用户密码错误");
    }
    return username;
}
  • 测试输出
    attribute
    arrtribute_test
对象封装参数
  • SpringMVC同时也支持将多个参数封装进类中(需类提供set方法),并将传入数据自动配置到类的属性中
@RequestMapping("/account/login")
@ResponseBody
public String accountLogin(Account account){
    System.out.println(account.getUserName()+
            "\n用户密码:"+account.getPassword());
    return "登录成功";
}
  • 测试输出
    objectobject_test

接收数组类型

  • 数组类型发送方式为:参数1=xxx&参数1=xxx参数1=xxx,SpringMVC会自动将参数1的数据转化为参数1数组中,数组可以是List也可以是数组[]
@RequestMapping("/cart")
@ResponseBody
public String addToCart(@RequestParam List<Integer> productIds){
    System.out.println(productIds);
    return productIds.toString();
}

list
list_test

Restful

  • Restful(Representational State Transfer) 是一种软件架构风格,用于设计网络应用程序,本身是基于一组原则和约束,用于提高系统的可伸缩性,简化通信,提高可见性并降低应用复杂性。其中Restful架构包含的关键约束有:

    • 无状态性(Statelessness):客户端每个请求都必须包含服务器处理所需的所有信息,服务器不应保存客户端的状态,每个请求都必须是独立的。
    • 基于资源(Resource):资源是Restful架构的核心,资源是应用程序中的实体或服务,如用户,订单,产品等,每个资源都应该有唯一的资源路径定位符(URL)
    • 表现层状态转化(Representational State Transfer):表现层是资源(数据或功能)当前状态的表现形式,客户端通过与资源的表现层交互来实现资源的状态转化,通常使用Json或XML作为表现层(简单点说就是使用Json数据与服务端进行交互通信)。
    • 统一接口(Uniform Interface):Restful应该有一个统一的接口,一提高系统的可见性。
    操作对应接口
    通过xx获取指定数据GET
    获取全部数据GET
    添加数据POST
    修改数据PUT
    删除数据DELETE
    • 按需可缓存性:Restful服务能够使用缓存来提高性能,服务器与客户端之间的通信应该标识哪些信息是可以缓存的,并在适当的时候使用缓存
    • 分层系统:构建Restful服务时使用分层系统,每层都提供特定的功能,每个层都只知道与其直接相邻的层的信息,从而提高了系统的可伸缩性(SpringMVC本身就是一个分层架构
  • 以下为Restful风格样例,@RestController里包含@RequestBody与@Controller,并且product有属于自己的资源访问路径/products,客户端与服务端交互使用json数据格式

@RestController
@RequestMapping("/products")
public class ProductController {
    @Autowired
    private ProductService productService;

    @PostMapping
    public Result saveProduct(@RequestBody Product product) {
        boolean flag = productService.saveProduct(product);
        return new Result(flag ? Code.SAVE_OK.getCode() : Code.SAVE_ERROR.getCode(), flag);
    }

    @PutMapping
    public Result updateProduct(@RequestBody Product product) {
        boolean flag = productService.updateProduct(product);
        return new Result(flag ? Code.UPDATE_OK.getCode() : Code.UPDATE_ERROR.getCode(), flag);
    }

    @DeleteMapping("/{id}")
    public Result deleteProduct(@PathVariable Integer id) {
        boolean flag = productService.deleteProductById(id);
        return new Result(flag ? Code.DELETE_OK.getCode() : Code.DELETE_ERROR.getCode(), flag);
    }

    @GetMapping("/{id}")
    public Result getProduct(@PathVariable Integer id) {
        Product product = productService.getProductById(id);
        boolean flag = (product != null);
        String msg = flag ? "数据查询成功" : "数据查询失败,请重试!";
        return new Result(flag ? Code.GET_OK.getCode() : Code.GET_ERROR.getCode(), product, msg);
    }

    @GetMapping
    public Result getAll() {
        List<Product> list = productService.getAllProducts();
        boolean flag = (list != null);
        String msg = flag ? "数据查询成功" : "数据查询失败,请重试!";
        return new Result(flag ? Code.GET_OK.getCode() : Code.GET_ERROR.getCode(), list, msg);
    }

}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1517364.html

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

相关文章

大模型笔记:吴恩达 ChatGPT Prompt Engineering for Developers(1) prompt的基本原则和策略

1 intro 基础大模型 VS 用指令tune 过的大模型 基础大模型 只会对prompt的文本进行续写 所以当你向模型发问的时候&#xff0c;它往往会像复读机一样续写几个问题这是因为在它见过的语料库文本&#xff08;通常大多来自互联网&#xff09;中&#xff0c;通常会连续列举出N个问…

Flask vs. Django:选择适合你的Web开发框架【第134篇—Flask vs. Django】

Flask vs. Django&#xff1a;选择适合你的Web开发框架 在选择一个适合你项目的Web开发框架时&#xff0c;常常会遇到 Flask 和 Django 这两个流行的选择。两者都有其优势和适用场景&#xff0c;本文将探讨它们的特点&#xff0c;并通过代码实例和解析来帮助你更好地做出选择。…

环形链表的起点——细节讲解

对于一个环形链表&#xff0c;我们要找到他的起点。可以通过如下推导。 我们设置两个快慢指针&#xff0c;相遇的点为X. 到起点的距离是T&#xff0c;圈长是C&#xff0c;第一次相交的点是X &#xff08;TX&#xff09;2TNCX 化出来TN*C-X 也就是说我们把一个节点放头部重新遍…

SQLiteC/C++接口详细介绍之sqlite3类(六)

快速前往文章列表&#xff1a;SQLite—系列文章目录 上一篇&#xff1a;SQLiteC/C接口详细介绍之sqlite3类&#xff08;五&#xff09; 下一篇&#xff1a;SQLiteC/C接口详细介绍之sqlite3类&#xff08;七&#xff09; 19. sqlite3_changes与sqlite3_changes64 是SQLite中用…

手机安装Kali Linux

在数字化时代&#xff0c;信息安全和隐私保护显得尤为重要。Kali Linux&#xff0c;作为一款专业的渗透测试和安全审计工具&#xff0c;因其强大的功能和丰富的资源库而受到广大安全研究者和爱好者的青睐。然而&#xff0c;我们通常只能在传统的电脑设备上安装和使用Kali Linux…

外卖小程序-购物车模块表结构设计和后端代码

表结构设计 添加购物车代码 Service public class ShoppingCartServiceImpl implements ShoppingCartService {Autowiredprivate ShoppingCartMapper shoppingCartMapper;Autowiredprivate DishMapper dishMapper;Autowiredprivate SetmealMapper setmealMapper;/*** 添加购物…

在Linux/Ubuntu/Debian中使用7z压缩和解压文件

要在 Ubuntu 上使用 7-Zip 创建 7z 存档文件&#xff0c;你可以使用“7z”命令行工具。 操作方法如下&#xff1a; 安装 p7zip&#xff1a; 如果你尚未在 Ubuntu 系统上安装 p7zip&#xff08;7-Zip 的命令行版本&#xff09;&#xff0c;你可以使用以下命令安装它&#xff1a;…

某夕夕商品数据抓取逆向之webpack扣取

逆向网址 aHR0cHM6Ly93d3cucGluZHVvZHVvLmNvbQ 逆向链接 aHR0cHM6Ly93d3cucGluZHVvZHVvLmNvbS9ob21lL2JveXNoaXJ0 逆向接口 aHR0cHM6Ly9hcGl2Mi5waW5kdW9kdW8uY29tL2FwaS9naW5kZXgvdGYvcXVlcnlfdGZfZ29vZHNfaW5mbw 逆向过程 请求方式&#xff1a;GET 参数构成 【anti_content】…

PHP中的反序列化漏洞

PHP中的反序列化漏洞 目录 PHP 中的序列化与反序列化 概述 序列化 基本类型的序列化 对象的序列化 反序列化 示例序列化与反序列化 反序列化漏洞 - PHP 中的魔术方法 - Typecho_v1.0 中的反序列化漏洞 POP链的构造思路 pop链案例 反序列化逃逸 字符串逃逸&#xff…

GoLang:云原生时代致力于构建高性能服务器的后端语言

Go语言的介绍 概念 Golang&#xff08;也被称为Go&#xff09;是一种编程语言&#xff0c;由Google于2007年开始设计和开发&#xff0c;并于2009年首次公开发布。Golang是一种静态类型、编译型的语言&#xff0c;旨在提供高效和可靠的软件开发体验。它具有简洁的语法、高效的编…

C# wpf 使用GDI实现截屏

wpf截屏系列 第一章 使用GDI实现截屏&#xff08;本章&#xff09; 第二章 使用GDI实现截屏 第三章 使用DockPanel制作截屏框 第四章 实现截屏框热键截屏 第五章 实现截屏框实时截屏 第六章 使用ffmpeg命令行实现录屏 文章目录 wpf截屏系列前言一、导入gdi32方法一、NuGet获取…

【LeetCode: 102. 二叉树的层序遍历 + bfs】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

国产化三防笔记本丨亿道国产加固笔记本FT-2000/4处理器

国产化加固笔记本是指采用国产操作系统和处理器&#xff0c;通过技术手段对其进行硬件加固、软件加密、数据安全等多方面加强处理的产品。这种笔记本电脑通常被用于政府项目、金融行业等对安全性要求极高的领域。 在国产化加固笔记本中&#xff0c;硬件加固是重要的一环。为了保…

架构实战--以海量存储系统讲解热门话题:分布式概念

关注我&#xff0c;持续分享逻辑思维&管理思维&#xff1b; 可提供大厂面试辅导、及定制化求职/在职/管理/架构辅导&#xff1b; 有意找工作的同学&#xff0c;请参考博主的原创&#xff1a;《面试官心得--面试前应该如何准备》&#xff0c;《面试官心得--面试时如何进行自…

腾讯云轻量服务器地域选择教程,2024最新地域选择攻略

腾讯云服务器地域怎么选择&#xff1f;不同地域之间有什么区别&#xff1f;腾讯云哪个地域好&#xff1f;地域选择遵循就近原则&#xff0c;访客距离地域越近网络延迟越低&#xff0c;速度越快。腾讯云百科txybk.com告诉大家关于地域的选择还有很多因素&#xff0c;地域节点选择…

Github主页设置贪吃蛇详细教程

先看最终实现结果&#xff1a; 有条贪吃蛇放在主页还是蛮酷的哈哈哈。接下来我来讲一讲怎么在Github主页添加一条贪吃蛇。 首先要修改自己的Github的主页&#xff0c;我们得有一个特殊的仓库——这个仓库必须与你的Github用户名保持一致&#xff0c;并且需要公开&#xff0c…

ArcGIS全系列实战视频教程——9个单一课程组合+系列直播回放

《ArcGIS全系列实战视频教程》是由9个单一课程组合合成。组成一条ArcGIS入门实战各项专题深入应用学习全链条&#xff0c;让你学有方向、学有目的&#xff0c;系统全面掌握ArcGIS。 ArcGIS全系列实战视频教程——9个单一课程组合https://edu.csdn.net/combo/detail/2569 《Ar…

unity2D生成9*9格子

1.创建一个空对象和格子 2将格子做成预制体&#xff08;直接将格子拖到这里即可&#xff0c;拖了过后删掉原来的格子&#xff09; 3.创建脚本并将脚本拖到空对象上 using System.Collections; using System.Collections.Generic; using UnityEngine;public class CreateMap : M…

增删卜易——八宫六十四卦

之前看倪海厦的《天纪》笔记里面提到了六十四卦世应,觉得不知道这个世应是啥意思。很长时间就没看了,偶然间看到了张文江教授写的一本书《潘雨廷先生谈话录》提到了《卜筮正宗》,“卜筮最后的判断是非理性转义,其他一切都只是形式”,“明人的著作,从京氏易出,如今天几日…

GitHub 服务器

GitHub 服务器 公司中&#xff0c;我们可以搭建中央服务器让项目组开发人员共享代码&#xff0c;但是如果我们的开发人员都是通过互联网进行协作&#xff0c;而不是在同一个地方&#xff0c;那么开发时&#xff0c;程序文件代码的版本管理就显得更加重要&#xff0c;这就需要搭…