首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >【愚公系列】2023年10月 Java教学课程 155-Spring MVC框架的REST风格API

【愚公系列】2023年10月 Java教学课程 155-Spring MVC框架的REST风格API

作者头像
愚公搬代码
发布2025-05-28 15:36:07
发布2025-05-28 15:36:07
15600
代码可运行
举报
文章被收录于专栏:历史专栏历史专栏
运行总次数:0
代码可运行

🚀一、REST风格

🔎1.REST简介

REST(Representational State Transfer)是一种软件架构风格,它的核心理念是将应用程序的功能转化为可以被URL所表示的资源,并通过HTTP协议对这些资源进行操作和交互。RESTful的优点包括可伸缩性、灵活性和可重用性。客户端不需要了解服务器的内部实现,只需要通过URI和HTTP方法即可操作服务器端的资源。

以下是一个简单的REST示例,假设我们要创建一个应用程序,用于管理书籍信息。我们可以使用RESTful架构来进行设计。

  1. 资源的定义及URI:

我们定义每一本书为一个资源,每个资源都有一个唯一的URI,例如:

代码语言:javascript
代码运行次数:0
运行
复制
/books/1234
  1. HTTP方法的定义:

我们使用HTTP方法来操作资源,如GET、POST、PUT、DELETE等方法。

  • GET方法用于获取资源信息。
  • POST方法用于创建新资源。
  • PUT方法用于更新或替换资源。
  • DELETE方法用于删除资源。
  1. HTTP状态码的定义:

HTTP状态码被用来表示资源和操作的结果,如200表示成功、404表示资源不存在、500表示服务器错误等。

例如,当创建新书时,如果操作成功,返回状态码201(Created)。如果书籍已存在,返回状态码409(Conflict)。如果请求的格式不正确,返回状态码400(Bad Request)等。

  1. 示例:

我们可以使用HTTP请求来操作书籍信息,例如:

  • 获取一本书:
代码语言:javascript
代码运行次数:0
运行
复制
GET /books/1234
  • 创建一本新书:
代码语言:javascript
代码运行次数:0
运行
复制
POST /books
{"title": "RESTful Web Services", "author": "Leonard Richardson"}
  • 更新一本书:
代码语言:javascript
代码运行次数:0
运行
复制
PUT /books/1234
{"title": "RESTful Web Services", "author": "Leonard Richardson"}
  • 删除一本书:
代码语言:javascript
代码运行次数:0
运行
复制
DELETE /books/1234

通过使用RESTful架构,我们可以快速、简单地进行Web应用程序开发,并且可以通过HTTP协议实现跨平台、跨语言的交互。

🔎2.RESTful入门案例

🦋2.1 快速入门

做法:在Controller中定义方法时设定"http请求动作(请求方式)“和"设定请求参数(路径变量)”

代码语言:javascript
代码运行次数:0
运行
复制
@Controller
public class UserController {

    //设置当前请求方法为POST,表示REST风格中的添加操作
    @RequestMapping(value = "/users",method = RequestMethod.POST)
    @ResponseBody
    public String save(){
        System.out.println("user save...");
        return "{'module':'user save'}";
    }

    //设置当前请求方法为DELETE,表示REST风格中的删除操作
    //@PathVariable注解用于设置路径变量(路径参数),要求路径上设置对应的占位符,并且占位符名称与方法形参名称相同
    @RequestMapping(value = "/users/{id}",method = RequestMethod.DELETE)
    @ResponseBody
    public String delete(@PathVariable Integer id){
        System.out.println("user delete..." + id);
        return "{'module':'user delete'}";
    }

    //设置当前请求方法为PUT,表示REST风格中的修改操作
    @RequestMapping(value = "/users",method = RequestMethod.PUT)
    @ResponseBody
    public String update(@RequestBody User user){
        System.out.println("user update..."+user);
        return "{'module':'user update'}";
    }

    //设置当前请求方法为GET,表示REST风格中的查询操作
    //@PathVariable注解用于设置路径变量(路径参数),要求路径上设置对应的占位符,并且占位符名称与方法形参名称相同
    @RequestMapping(value = "/users/{id}" ,method = RequestMethod.GET)
    @ResponseBody
    public String getById(@PathVariable Integer id){
        System.out.println("user getById..."+id);
        return "{'module':'user getById'}";
    }

    //设置当前请求方法为GET,表示REST风格中的查询操作
    @RequestMapping(value = "/users",method = RequestMethod.GET)
    @ResponseBody
    public String getAll(){
        System.out.println("user getAll...");
        return "{'module':'user getAll'}";
    }
}
🦋2.2 @PathVariable介绍
  • 名称:@PathVariable
  • 类型:形参注解
  • 位置:SpringMVC控制器方法形参定义前面
  • 作用:绑定路径参数与处理器方法形参间的关系,要求路径参数名与形参名一一对应
🦋2.3 @RequestBody、@RequestParam、@PathVariable区别和应用
  • 区别 @RequestParam用于接收url地址传参或表单传参 @RequestBody用于接收json数据 @PathVariable用于接收路径参数,使用{参数名称}描述路径参数
  • 应用 后期开发中,发送请求参数超过1个,以json格式为主,@RequestBody应用较广 如果发送非json格式数据,选用@RequestParam接收请求参数 采用RESTful进行开发,当参数数量较少时,例如1个,可以采用@PathVariable接收请求路径变量,通常用于传递id值

🔎3.REST快速开发【重点】

🦋3.1 代码中的问题

以上截图中的代码和我们之前写的UserController中的方法类似,其中图中两个方法都有三处是有问题的,可以进行优化。存在的问题如下:

问题1:每个方法的@RequestMapping注解中都定义了访问路径/books,重复性太高。

问题2:每个方法的@RequestMapping注解中都要使用method属性定义请求方式,重复性太高。

问题3:每个方法响应json都需要加上@ResponseBody注解,重复性太高。

🦋3.2 Rest快速开发

解决以上三个问题

解决问题1:在Controller类上使用@RequestMapping定义共同的访问路径。

代码语言:javascript
代码运行次数:0
运行
复制
@Controller
@RequestMapping("/books")
public class BookController {
    
    @RequestMapping(method = RequestMethod.POST)
    public String save(@RequestBody Book book){
        System.out.println("book save..." + book);
        return "{'module':'book save'}";
    }
    @RequestMapping(value = "/{id}" ,method = RequestMethod.DELETE)
    public String delete(@PathVariable Integer id){
        System.out.println("book delete..." + id);
        return "{'module':'book delete'}";
    }
    @RequestMapping(method = RequestMethod.PUT)
    public String update(@RequestBody Book book){
        System.out.println("book update..."+book);
        return "{'module':'book update'}";
    }
    @RequestMapping(value = "/{id}" ,method = RequestMethod.GET)
    public String getById(@PathVariable Integer id){
        System.out.println("book getById..."+id);
        return "{'module':'book getById'}";
    }

    @RequestMapping(method = RequestMethod.GET)
    public String getAll(){
        System.out.println("book getAll...");
        return "{'module':'book getAll'}";
    }
}

解决问题2:使用@GetMapping @PostMapping @PutMapping @DeleteMapping代替@RequestMapping(method=RequestMethod.XXX)

代码语言:javascript
代码运行次数:0
运行
复制
@Controller   
@RequestMapping("/books")
public class BookController {

//    @RequestMapping( method = RequestMethod.POST)
    @PostMapping//使用@PostMapping简化Post请求方法对应的映射配置
    public String save(@RequestBody Book book){
        System.out.println("book save..." + book);
        return "{'module':'book save'}";
    }

//    @RequestMapping(value = "/{id}" ,method = RequestMethod.DELETE)
    @DeleteMapping("/{id}")  //使用@DeleteMapping简化DELETE请求方法对应的映射配置
    public String delete(@PathVariable Integer id){
        System.out.println("book delete..." + id);
        return "{'module':'book delete'}";
    }

//    @RequestMapping(method = RequestMethod.PUT)
    @PutMapping   //使用@PutMapping简化Put请求方法对应的映射配置
    public String update(@RequestBody Book book){
        System.out.println("book update..."+book);
        return "{'module':'book update'}";
    }

//    @RequestMapping(value = "/{id}" ,method = RequestMethod.GET)
    @GetMapping("/{id}")    //使用@GetMapping简化GET请求方法对应的映射配置
    public String getById(@PathVariable Integer id){
        System.out.println("book getById..."+id);
        return "{'module':'book getById'}";
    }

//    @RequestMapping(method = RequestMethod.GET)
    @GetMapping      //使用@GetMapping简化GET请求方法对应的映射配置
    public String getAll(){
        System.out.println("book getAll...");
        return "{'module':'book getAll'}";
    }
}
  • 名称:@GetMapping @PostMapping @PutMapping @DeleteMapping
  • 类型:方法注解
  • 位置:基于SpringMVC的RESTful开发控制器方法定义上方
  • 作用:设置当前控制器方法请求访问路径与请求动作,每种对应一个请求动作,例如@GetMapping对应GET请求
  • 属性: value(默认):请求访问路径

解决问题3:在Controller类上使用@RestController注解,等同于@Controller与@ResponseBody两个注解组合功能

代码语言:javascript
代码运行次数:0
运行
复制
@RestController     //使用@RestController注解替换@Controller与@ResponseBody注解,简化书写
@RequestMapping("/books")
public class BookController {
    //方法省略了没写
}
  • 名称:@RestController
  • 类型:类注解
  • 位置:基于SpringMVC的RESTful开发控制器类定义上方
  • 作用:设置当前控制器类为RESTful风格,等同于@Controller与@ResponseBody两个注解组合功能

🔎4.案例:基于RESTful页面数据交互

🦋4.1 案例效果和环境准备
☀️4.1.1 案例效果
☀️4.1.2 环境准备

pom.xml

代码语言:javascript
代码运行次数:0
运行
复制
<dependencies>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-webmvc</artifactId>
        <version>5.2.10.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.projectlombok</groupId>
        <artifactId>lombok</artifactId>
        <version>1.18.16</version>
        <scope>provided</scope>
    </dependency>

    <dependency>
        <groupId>javax.servlet</groupId>
        <artifactId>javax.servlet-api</artifactId>
        <version>3.1.0</version>
        <scope>provided</scope>
    </dependency>

    <!-- 要导入jackson的jar包,才能转换 -->
    <dependency>
        <groupId>com.fasterxml.jackson.core</groupId>
        <artifactId>jackson-databind</artifactId>
        <version>2.11.3</version>
    </dependency>
</dependencies>

代码

代码语言:javascript
代码运行次数:0
运行
复制
@Data
public class Book {
    private Integer id;
    private String type;
    private String name;
    private String description;
}
代码语言:javascript
代码运行次数:0
运行
复制
//SpringMVC容器初始化类
public class ServletConfigInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {
    protected Class<?>[] getRootConfigClasses() {
        return new Class[0];
    }

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

    protected String[] getServletMappings() {
        return new String[]{"/"};
    }
    //乱码处理
    @Override
    protected Filter[] getServletFilters() {
        CharacterEncodingFilter filter = new CharacterEncodingFilter();
        filter.setEncoding("UTF-8");
        return new Filter[]{filter};
    }
}
代码语言:javascript
代码运行次数:0
运行
复制
//SpringMVC配置类
@Configuration
//同时扫描controller和config两个包
@ComponentScan({"com.itheima.controller","com.itheima.config"})
@EnableWebMvc
public class SpringMvcConfig {
}
🦋4.2 代码实现
☀️4.2.1 制作SpringMVC控制器,并通过PostMan测试接口功能
代码语言:javascript
代码运行次数:0
运行
复制
@RestController
@RequestMapping("/books")
public class BookController {

    @PostMapping
    public String save(@RequestBody Book book){
        System.out.println("book save ==> "+ book);
        return "success";
    }

    @GetMapping
    public List<Book> getAll(){
        System.out.println("book getAll is running ...");
        List<Book> bookList = new ArrayList<>();

        Book book1 = new Book();
        book1.setId(1);
        book1.setType("计算机");
        book1.setName("SpringMVC入门教程");
        book1.setDescription("小试牛刀");
        bookList.add(book1);

        Book book2 = new Book();
        book2.setId(2);
        book2.setType("计算机");
        book2.setName("SpringMVC实战教程");
        book2.setDescription("一代宗师");
        bookList.add(book2);

        Book book3 = new Book();
        book3.setId(3);
        book3.setType("计算机丛书");
        book3.setName("SpringMVC实战教程进阶");
        book3.setDescription("一代宗师呕心创作");
        bookList.add(book3);

        return bookList;
    }
}

查询

添加

代码语言:javascript
代码运行次数:0
运行
复制
{
    "id":5,
    "type":"小说",
    "name": "西游记",
    "description": "四大名著之一"
}

控制台输出

代码语言:javascript
代码运行次数:0
运行
复制
book save ==> Book(id=5, type=小说, name=西游记, description=四大名著之一)
☀️4.2.2 设置对静态资源的访问放行
  1. 复制页面到webapp目录下
  2. 对静态资源放行

在config包编写类继承于WebMvcConfigurationSupport,中重写addResourceHandlers方法,在类上添加@Configuration注解

  1. addResourceHandler映射的地址,/**表示包含子孙目录
  2. addResourceLocations表示物理存在的地址
  3. 四个静态目录都进行了映射
代码语言:javascript
代码运行次数:0
运行
复制
@Configuration
public class SpringMvcSupport extends WebMvcConfigurationSupport {
    //设置静态资源访问过滤,当前类需要设置为配置类,并被扫描加载
    @Override
    protected void addResourceHandlers(ResourceHandlerRegistry registry) {
        //当访问/pages/xxxx时候,从/pages目录下查找内容
        registry.addResourceHandler("/pages/**")
            .addResourceLocations("/pages/");
        registry.addResourceHandler("/js/**")
            .addResourceLocations("/js/");        		
        registry.addResourceHandler("/css/**")
            .addResourceLocations("/css/");       
        registry.addResourceHandler("/plugins/**")
            .addResourceLocations("/plugins/");
    }
}
  1. 修改SpringMvcConfig中扫描config包
代码语言:javascript
代码运行次数:0
运行
复制
@Configuration
@ComponentScan({"com.itheima.controller", "com.itheima.config"})
@EnableWebMvc
public class SpringMvcConfig {
}
☀️4.2.3 前端页面通过异步提交访问后台控制器
代码语言:javascript
代码运行次数:0
运行
复制
//添加
saveBook () {
    axios.post("/books",this.formData).then((res)=>{
        if (res.data=="success") {
            this.$message.success("添加成功");
        }
    });
},

//主页列表查询
getAll() {
    axios.get("/books").then((res)=>{
        this.dataList = res.data;
    });
},
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2025-05-16,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 🚀一、REST风格
    • 🔎1.REST简介
    • 🔎2.RESTful入门案例
      • 🦋2.1 快速入门
      • 🦋2.2 @PathVariable介绍
      • 🦋2.3 @RequestBody、@RequestParam、@PathVariable区别和应用
    • 🔎3.REST快速开发【重点】
      • 🦋3.1 代码中的问题
      • 🦋3.2 Rest快速开发
    • 🔎4.案例:基于RESTful页面数据交互
      • 🦋4.1 案例效果和环境准备
      • 🦋4.2 代码实现
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档