SpringMVC修炼之旅(2)基础入门

一、第一个程序

1.1环境配置

1.2代码实现

java 复制代码
package com.itheima.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

//定义表现层控制器bean
@Controller
public class UserController {

    //设置映射路径为/save,即外部访问路径
    @RequestMapping("/save")
    //设置当前操作返回结果为指定json数据(本质上是一个字符串信息)
    @ResponseBody
    public String save(){
        System.out.println("user save ...");
        return "{'info':'springmvc'}";
    }

    //设置映射路径为/delete,即外部访问路径
    @RequestMapping("/delete")
    @ResponseBody
    public String delete(){
        System.out.println("user save ...");
        return "{'info':'springmvc'}";
    }
}

1.3注解解析

@Controller

类型:类注解

位置:SpringMVC控制器类定义上方

作用:用于标记控制器类。当一个类被@Controller注解标记时,Spring会自动将该类识别为一个控制器类,并负责处理与该类相关的HTTP请求。

java 复制代码
@Controller
public class UserController 
{

}

@RequestMapping

类型:方法注解

位置:SpringMVC控制器方法定义上方

作用:用于映射HTTP请求到特定的处理方法。它可以应用于类、方法或参数上。当一个类被@RequestMapping注解标记时,Spring会自动将该类中的所有带有@RequestMapping注解的方法识别为处理特定URL路径的处理器。:

java 复制代码
@Controller
public class UserController {

    @RequestMapping(value = "/save")
    public void save() {
        System.out.println("user save ...");
    }
}

相关属性 value(默认):请求访问路径

method:http请求动作,标准动作(GET/POST/PUT/DELETE)

@ResponseBody

类型:方法注解

位置:SpringMVC控制器方法定义上方

作用:用于将控制器方法的返回值直接写入HTTP响应体中。这样,当客户端发起请求时,服务器端会将处理结果以指定的格式(如JSON、XML等)发送给客户端

1.4bean加载控制

Spring控制的bean

  • 业务bean(Service)
  • 功能bean(DataSource等)

@Configuration

`@Configuration`注解是Spring框架中的一个注解,用于标记一个类作为配置类。配置类通常用于定义Bean的创建和依赖关系,以便在应用程序中进行依赖注入。

@ComponentScan

类型:类注解

位置:应该放在Spring配置文件的类上,通常是在配置类中

作用:用于指定要扫描的包路径,以便自动注册该包及其子包中的组件。当使用@ComponentScan注解时,Spring会自动扫描指定的包路径下的所有类,并将带有@Component、@Service、@Repository或@Controller等注解的类注册为Spring容器中的Bean。

java 复制代码
@Configuration
@ComponentScan(value = "com.com.example",
    excludeFilters = @ComponentScan.Filter(
        type = FilterType.ANNOTATION,
        classes = Controller.class
    )
)
public class SpringConfig {
}

excludeFilters:排除扫描路径中加载的bean,需要指定类别(type)与具体项(classes) includeFilters:加载指定的bean,需要指定类别(type)与具体项(classes)

@EnableAutoConfiguration

是Spring Boot框架提供的一个注解,用于自动配置Spring应用程序的配置。它会根据应用程序的依赖和配置自动配置Spring应用程序的配置。这样可以减少开发人员的配置工作,提高开发效率。@EnableAutoConfiguration注解通常与@SpringBootApplication注解一起使用,用于启用Spring Boot自动配置功能。

@SpringBootApplication

`@SpringBootApplication`注解是Spring Boot框架中的一个注解,用于标记`@SpringBootApplication`注解是Spring Boot框架中的一个注解,用于标记一个类作为Spring Boot应用程序的主入口。它包含了`@Configuration`、`@EnableAutoConfiguration`和`@ComponentScan`三个注解的组合,可以自动配置Spring Boot应用程序的组件扫描路径、自动配置和其他相关设置。

二、请求和响应

2.1请求映射路径

多个控制器类处理相同的请求路径

可以通过以下方法解决

使用@RequestMapping注解的value属性指定请求路径的具体值,可以在不同的控制器类中使用不同的请求路径来区分不同的处理方法。

java 复制代码
@Controller
@RequestMapping("/example")
public class ExampleController1 {
    @RequestMapping("/test")
    public String test() {
        //处理逻辑
        return "test";
    }
}

@Controller
@RequestMapping("/example2")
public class ExampleController2 {
    @RequestMapping("/test")
    public String test() {
        //处理逻辑
        return "test2";
    }
}

使用@RequestMapping注解的params属性来区分不同的请求处理方法,可以根据请求参数的不同来调用不同的处理方法。

java 复制代码
@Controller
@RequestMapping("/example")
public class ExampleController {
    @RequestMapping(value = "/test", params = "param1")
    public String test1() {
        //处理逻辑
        return "test1";
    }

    @RequestMapping(value = "/test", params = "param2")
    public String test2() {
        //处理逻辑
        return "test2";
    }
}

使用@RequestMapping注解的headers属性来区分不同的请求处理方法,可以根据请求头的不同来调用不同的处理方法。

java 复制代码
@Controller
@RequestMapping("/example")
public class ExampleController {
    @RequestMapping(value = "/test", headers = "header1=value1")
    public String test1() {
        //处理逻辑
        return "test1";
    }

    @RequestMapping(value = "/test", headers = "header2=value2")
    public String test2() {
        //处理逻辑
        return "test2";
    }
}

2.2请求方式

  • Get请求
  • Post请求

Get请求传参

普通参数:url地址传参,地址参数名与形参变量名相同,定义形参即可接收参数

java 复制代码
@RequestMapping("/commonParam")
    @ResponseBody
    public String handleCommonParam(String name,Integer age) {
        System.out.println("Name: " + name + ", age: " + age);
        return "{'module':'common param'}";
    }
bash 复制代码
GET http://localhost:8080/commonParam?name=11&age=11

Post请求参数

java 复制代码
POST http://localhost:8080/commonParam
Content-Type: application/x-www-form-urlencoded

name=11&age=18

Get请求和 Post请求的区别

GET和POST是HTTP协议中两种最常被使用的请求方法,它们在原理、使用、速度和安全性等方面都有所区别。具体来说:

  1. 参数传递方式:GET方法将参数包含在URL中,而POST方法通过request body传递参数。这意味着,使用GET提交的数据会显示在URL上,而POST提交的数据则不会显示在URL上。

  2. 数据用途:GET请求通常用于从服务器获取数据,而POST请求则主要用于向服务器提交数据,例如提交表单数据。

  3. 缓存与历史记录:GET请求可以被浏览器缓存,同时也会保存在浏览器的历史记录中。相反,POST请求不会被缓存,且不会保存在历史记录中。

  4. 安全性:由于GET请求将参数直接暴露在URL中,因此可能存在安全隐患。相反,POST请求通过加密的request body传递参数,因此在安全性方面相对较高。

  5. 数据量限制:GET方法由于数据量受限于URL的长度,因此对于大量数据的提交不太适用。而POST方法没有这方面的限制。

  6. 影响刷新和回退:当使用GET请求时,刷新页面或回退操作不会影响之前的数据。但是,使用POST请求时,回退可能会导致数据被重新提交。

2.3请求参数

参数种类

  • 普通参数
  • POJO类型参数
  • 嵌套POJO类型参数
  • 数组类型参数
  • 集合类型参数

普通参数:

  • url地址传参,地址参数名与形参变量名相同,定义形参即可接收参数
  • 请求参数名与形参变量名不同,使用@RequestParam绑定参数关系
java 复制代码
 @RequestMapping("/commonParam")
    @ResponseBody
    public String handleCommonParam(@RequestParam("name") String name1,Integer age) {
        System.out.println("Name: " + name1 + ", age: " + age);
        return "{'module':'common param'}";
    }

@RequestParam 注解

类型:形参注解

位置:SpringMVC控制器方法形参定义前面

作用:用于将请求参数绑定到方法的参数上。它可以用于处理 GET 和 POST 请求中的查询参数和表单数据。使用 @RequestParam 注解可以将请求参数的值绑定到方法的参数上

它有以下两个参数:

  1. required:布尔类型,表示该参数是否为必传参数。如果设置为 true,则在请求中必须包含该参数;如果设置为 false,则可以不包含该参数,此时参数的默认值为 null

  2. defaultValue:字符串类型,表示当请求中没有提供该参数时使用的默认值。如果不设置此参数,或者设置为空字符串,那么在请求中没有提供该参数时,参数的值将为 null

java 复制代码
@RequestMapping("/commonParam")
    @ResponseBody
    public String handleCommonParam(@RequestParam(value = "name", required = false, defaultValue = "defaultValue1") String name1,Integer age) {
        System.out.println("Name: " + name1 + ", age: " + age);
        return "{'module':'common param'}";
    }

POJO参数:

请求参数名与形参对象属性名相同,定义POJO类型形参即可接收参数

pojo类

java 复制代码
@RequestMapping("/pojoParam")
    @ResponseBody
    public String pojoParam(User user){
        System.out.println("pojo参数传递 user ==> "+user);
        return "{'module':'pojo param'}";
    }

嵌套POJO参数:

POJO对象中包含POJO对象

请求参数名与形参对象属性名相同,按照对象层次结构关系即可接收嵌套POJO属性参数

java 复制代码
 @RequestMapping("/pojoContainPojoParam")
    @ResponseBody
    public String pojoContainPojoParam(User user){
        System.out.println("pojo嵌套pojo参数传递 user ==> "+user);
        return "{'module':'pojo contain pojo param'}";
    }
bash 复制代码
GET http://localhost:8080//pojoContainPojoParam?name1=11&age=11&Address.province=12&Address.city=12

数组参数:

同名请求参数可以直接映射到对应名称的形参数组对象中

java 复制代码
//数组参数:同名请求参数可以直接映射到对应名称的形参数组对象中
    @RequestMapping("/arrayParam")
    @ResponseBody
    public String arrayParam(String[] likes){
        System.out.println("数组参数传递 likes ==> "+ Arrays.toString(likes));
        return "{'module':'array param'}";
    }
bash 复制代码
GET http://localhost:8080/arrayParam?likes=1&likes=2

集合保存普通参数:

请求参数名与形参集合对象名相同且请求参数为多个,@RequestParam绑定参数关系

java 复制代码
 //集合参数:同名请求参数可以使用@RequestParam注解映射到对应名称的集合对象中作为数据
    @RequestMapping("/listParam")
    @ResponseBody
    public String listParam(@RequestParam List<String> likes){
        System.out.println("集合参数传递 likes ==> "+ likes);
        return "{'module':'list param'}";
    }

集合参数:json格式

java 复制代码
 @RequestMapping("/listParamForJson")
    @ResponseBody
    public String listParamForJson(@RequestBody List<String> likes){
        System.out.println("list common(json)参数传递 list ==> "+likes);
        return "{'module':'list common for json param'}";
    }
java 复制代码
POST http://localhost:8080//listParamForJson
Content-Type: application/json

["like1", "like2", "like3"]

POJO参数:json数据

json数据与形参对象属性名相同,定义POJO类型形参即可接收参数

java 复制代码
@RequestMapping("/pojoParamForJson")
    @ResponseBody
    public String pojoParamForJson(@RequestBody User user){
        System.out.println("pojo(json)参数传递 user ==> "+user);
        return "{'module':'pojo for json param'}";
    }
bash 复制代码
POST http://localhost:8080/pojoParamForJson
Content-Type: application/json

{
"name": "John Doe", "age": 25,"address": {"province": "Ontario", "city": "Toronto"}
}

POJO集合参数:json

json数组数据与集合泛型属性名相同,定义List类型形参即可接收参数

java 复制代码
//集合参数:json格式
    //1.开启json数据格式的自动转换,在配置类中开启@EnableWebMvc
    //2.使用@RequestBody注解将外部传递的json数组数据映射到形参的保存实体类对象的集合对象中,要求属性名称一一对应
    @RequestMapping("/listPojoParamForJson")
    @ResponseBody
    public String listPojoParamForJson(@RequestBody List<User> list){
        System.out.println("list pojo(json)参数传递 list ==> "+list);
        return "{'module':'list pojo for json param'}";
    }
bash 复制代码
POST http://localhost:8080/listPojoParamForJson
Content-Type: application/json

[
  {
    "name": "John Doe", "age": 25,"address": {"province": "Ontario", "city": "Toronto"}
  },
  {
    "name": "John Doe", "age": 25,"address": {"province": "Ontario", "city": "Toronto"}
  }
]

@RequestBody与@RequestParam区别

区别

  • @RequestParam用于接收url地址传参,表单传参【application/x-www-form-urlencoded】
  • @RequestBody用于接收json数据【application/json】

应用

  • 后期开发中,发送json格式数据为主,@RequestBody应用较广
  • 如果发送非json格式数据,选用@RequestParam接收请求参数

日期类型参数传递

java 复制代码
//日期参数
//使用@DateTimeFormat注解设置日期类型数据格式,默认格式yyyy/MM/dd
@RequestMapping("/dataParam")
@ResponseBody
public String dataParam(Date date,
                        @DateTimeFormat(pattern="yyyy-MM-dd") Date date1,
                        @DateTimeFormat(pattern="yyyy/MM/dd HH:mm:ss") Date date2){
    System.out.println("参数传递 date ==> "+date);
    System.out.println("参数传递 date1(yyyy-MM-dd) ==> "+date1);
    System.out.println("参数传递 date2(yyyy/MM/dd HH:mm:ss) ==> "+date2);
    return "{'module':'data param'}";
bash 复制代码
GET http://localhost:8080/dataParam?date1=2023-12-06&date2=2023/12/06 10:10:10

想要将日期作为查询参数传递,因此你应该使用GET请求而不是POST请求。此外,由于日期值需要按照指定的格式进行设置,你需要对日期2的值进行编码,因为URL中的空格是不允许的

2.4响应

响应页面

java 复制代码
@RequestMapping("/toJumpPage")
public String toJumpPage(){
    System.out.println("跳转页面");
    return "redirect:/page.jsp";
}

响应数据

  • 文本数据
  • json数据
文本数据
java 复制代码
//响应文本数据
    //返回值为String类型,设置返回值为任意字符串信息,即可实现返回指定字符串信息,需要依赖@ResponseBody注解
    @RequestMapping("/toText")
    @ResponseBody
    public String toText(){
        System.out.println("返回纯文本数据");
        return "response text";
    }
json数据
java 复制代码
 @RequestMapping("/toJsonPOJO")
    @ResponseBody
    public User toJsonPOJO(){
        System.out.println("返回json对象数据");
        User user = new User();
        user.setName("itcast");
        user.setAge(15);
        return user;
    }
java 复制代码
 @RequestMapping("/toJsonList")
    @ResponseBody
    public List<User> toJsonList(){
        System.out.println("返回json集合数据");
        User user1 = new User();
        user1.setName("烟雨");
        user1.setAge(15);

        User user2 = new User();
        user2.setName("平生");
        user2.setAge(12);

        List<User> userList = new ArrayList<User>();
        userList.add(user1);
        userList.add(user2);

        return userList;
    }

HttpMessageConverter接口

java 复制代码
// 定义一个名为HttpMessageConverter的接口,泛型参数为T
public interface HttpMessageConverter<T> {
    // 判断是否可以读取指定类型的数据,参数为类对象和媒体类型(可以为空)
    boolean canRead(Class<?> clazz, @Nullable MediaType mediaType);
    // 判断是否可以写入指定类型的数据,参数为类对象和媒体类型(可以为空)
    boolean canWrite(Class<?> clazz, @Nullable MediaType mediaType);
    // 获取支持的媒体类型列表
    List<MediaType> getSupportedMediaTypes();
    // 从输入消息中读取指定类型的数据,参数为类对象和输入消息对象
    T read(Class<? extends T> clazz, HttpInputMessage inputMessage)
        throws IOException, HttpMessageNotReadableException;
    // 将指定类型的数据写入输出消息,参数为数据对象、内容类型(可以为空)和输出消息对象
    void write(T t, @Nullable MediaType contentType, HttpOutputMessage outputMessage)
        throws IOException, HttpMessageNotWritableException;
}
相关推荐
暮乘白帝过重山36 分钟前
Singleton和Prototype的作用域与饿汉式/懒汉式的初始化方式
spring·原型模式·prototype·饿汉式·singleton·懒汉式
ejinxian1 小时前
Spring AI Alibaba 快速开发生成式 Java AI 应用
java·人工智能·spring
杉之1 小时前
SpringBlade 数据库字段的自动填充
java·笔记·学习·spring·tomcat
圈圈编码2 小时前
Spring Task 定时任务
java·前端·spring
爱的叹息2 小时前
Java 连接 Redis 的驱动(Jedis、Lettuce、Redisson、Spring Data Redis)分类及对比
java·redis·spring
松韬3 小时前
Spring + Redisson:从 0 到 1 搭建高可用分布式缓存系统
java·redis·分布式·spring·缓存
天上掉下来个程小白3 小时前
Redis-14.在Java中操作Redis-Spring Data Redis使用方式-操作列表类型的数据
java·redis·spring·springboot·苍穹外卖
汤姆大聪明4 小时前
Redisson 操作 Redis Stream 消息队列详解及实战案例
redis·spring·缓存·maven
正经摸鱼5 小时前
classpath与classpath*实现逻辑
后端·spring
良枫5 小时前
Spring Security认证授权深度解析
spring boot·spring