小黑子—springMVC:第一章 请求处理与响应数据

news2024/11/29 8:50:31

springMVC入门1.0

  • 1、小黑子的springMVC基础
    • 1.1 SpringMVC概述
    • 1.2 SpringMVC快速入门
    • 1.3 Controller中直接注入spring中维护的Bean
    • 1.4 SpringMVC关键组件浅析
  • 2、SpringMVC的请求处理
    • 2.1 请求映射路径配置
    • 2.2 请求数据的接收
      • 2.2.1 键值对方式接收数据
        • 2.2.1 - I @RquestParam属性
      • 2.2.2 封装JavaBean数据
      • 2.2.3 请求参数Postman使用
      • 2.2.4 接收JSON格式数据
      • 2.2.5 接收Restful风格数据
      • 2.2.6 接收上传的文件
      • 2.2.6 获取header头信息和cookie信息
      • 2.2.7 直接获取Requset和Session域中的数据
    • 2.3 Javaweb常用对象获取
    • 2.4 请求静态资源
    • 2.5 注解驱动< mvc:annotation-driven >标签
  • 3、SpringMVC的响应处理
    • 3.1、传统同步业务数据响应
    • 3.2 前后端分离异步业务数据响应

1、小黑子的springMVC基础

1.1 SpringMVC概述

SpringMVC是一个基于Spring开发的MVC轻量级框架,Spring3.0后发布的组件,SpringMVC和Spring可以无缝整合,使用DispatcherServlet作为前端控制器,且内部提供了处理器映射器、处理器适配器、视图解析器等组件,可以简化JavaBean封装,Json转化、文件上传等操作。
在这里插入图片描述

1.2 SpringMVC快速入门

在这里插入图片描述
在创建Mave项目时要想配置运行tomact,从原型创建具体参考小黑子的第3章tomcat

  1. 导入spring-mvc坐标

    <dependency>
    	<groupId>org.springframework</groupId>
    	<artifactId>spring-webmvc</artifactId>
    	<version>5.3.7</version>
    </dependency>
    
  2. 配置前端控制器DispatcherServlet

    <?xml version="1.0" encoding="UTF-8"?>
    <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0">
    
     <!--    配置DispatcherServlet-->
      <servlet>
    <servlet-name>DispatcherServlet</servlet-name>
    	 <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    	<init-param>
      	<param-name>contextConfigLocation</param-name>
      	<param-value>classpath:spring-mvc.xml</param-value>
    	</init-param>
     <load-on-startup>2</load-on-startup>
    </servlet>
    
      <servlet-mapping>
    	<servlet-name>DispatcherServlet</servlet-name>
    <url-pattern>/</url-pattern>
     </servlet-mapping>
    
    </web-app>
    
  3. 编写Controller,配置映射路径,并交给SpringMVC容器管理

    package com.itheima.controller;
    
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.RequestMapping;
    
    @Controller
    public class QuickController {
    
    @RequestMapping("/show")
    public void show(){
        System.out.println("show 实现。。。。。");
    	}
    }
    

在这里插入图片描述
虽然页面报错,但是后台执行了show方法
在这里插入图片描述

页面报错500原因:视图指定

@Controller
public class QuickController {

    @RequestMapping("/show")
    public String show(){
        System.out.println("show 实现。。。。。");
        return "/index.jsp";
    }
}
<%@ page contentType="text/html;charset=UTF-8" language="java" %>

<html>
<body>
<h2>Hello SpringMVC!</h2>
</body>
</html>

在这里插入图片描述

1.3 Controller中直接注入spring中维护的Bean

  1. 创建Bean
@Service
public class QuickServiceImpl implements QuickService {
}
package com.itheima.service.impl;

import com.itheima.service.QuickService;
import org.springframework.stereotype.Service;

@Service
public class QuickServiceImpl implements QuickService {
}

  1. 用spring-xml文件扫描包含该Bean的包
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">

<!--    组件扫描-->
    <context:component-scan base-package="com.itheima.service"/>

</beans>
  1. 在Web-xml文件中配置ContextLoaderListener
<!--  配置一个初始化参数-->
  <context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath:applicationContext.xml</param-value>
  </context-param>
<!--  配置ContextLoaderListener-->
  <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
  1. 在Controller中访问容器中的Bean
@Controller
public class QuickController {
    //直接注入Service进行使用
    @Autowired
    private QuickService quickService;

    @RequestMapping("/show")
    public String show(){
        System.out.println("show 实现。。。。。"+quickService);
        return "/index.jsp";
    }
}

在这里插入图片描述

1.4 SpringMVC关键组件浅析

上面已经完成的快速入门的操作,也在不知不觉中完成的Spring和SpringMVC的整合,我们只需要按照规则去定义Controller和业务方法就可以。但是在这个过程中,肯定是很多核心功能类参与到其中,这些核心功能类,一般称为组件。当请求到达服务器时,是哪个组件接收的请求,是哪个组件帮我们找到的Controller,是哪个组件帮我们调用的方法,又是哪个组件最终解析的视图?/

组件描述常用组件
处理器映射器:HandlerMapping匹配映射路径对应的Handler,返回可执行的处理器链对象HandlerExecutionChain对象 RequestMappingHandlerMapping
处理器适配器:HandlerAdapter匹配HandlerExecutionChain对应的适配器进行处理器调用,返回视图模型对象RequestMappingHandlerAdapter
视图解析器:ViewResolver对视图模型对象进行解析InternalResourceViewResolver

在这里插入图片描述
SpringMVC加载组件的策略

SpringMVC的默认组件,SpringMVC在前端控制器DispatcherServlet加载时,就会进行初始化操作,在进行初始化时,就会加载SpringMVC默认指定的一些组件,这些默认组件配置在DispatcherServlet.properties文件中,该文件存在与spring-webmvc-5.3.7.jar包下的 org\springframework\weblservlet\DispatcherServlet.properties

org.springframework.web.servlet.HandlerMapping=org.springframewor.web.servlet.handler.BeanNameUrl HandlerMapping,\
	org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping,\
	org.springframework.web.servlet.function.support.RouterFunctionMapping
	
org.springframework.web.servlet.HandlerAdapter=org.springframework.web.servlet.mvc.HttpReguestHandlerAdapter,\
	org.springframework.web.servlet.mvc.SimpleControllerHaidlerAdapter,\
	org.springframework.web.servlet.mve.method.annotation.RequestMappingHandlerAdapter, 		 
	org.springframework.web.servlet.function.support.HandlerrunctionAdapter

org.springframework.web.servlet.VviewResolver=org.springframework.web.servlet.view.InternalResourceViewResolver

2、SpringMVC的请求处理

2.1 请求映射路径配置

配置映射路径,映射器处理器才能找到Controller的方法资源,目前主流映射路径配置方式就是@RequestMapping

相关注解作用使用位置
@RequestMapping设置控制器方法的访问资源路径,可以接收任何请求方法和类上
@GetMapping设置控制器方法的访问资源路径,可以接收GET请求方法和类上
@PostMapping设置控制器方法的访问资源路径,可以接收POST请求方法和类上
  • 当给类加了@RequestMapping,想要访问某个方法就必须要在虚拟地址上加入类的RequestMapping名以及方法的RequestMapping名
@Controller
@RequestMapping("/quick")
public class QuickController {
    //直接注入Service进行使用
    @Autowired
    private QuickService quickService;

//    @RequestMapping(value={"/show","showxxx","showyyy"},method = RequestMethod.GET)
//    @PostMapping("/show")

    //http://localhost:8080/appName/quick/show
    @GetMapping("/show")
    public String show(){
        System.out.println("show 实现。。。。。"+quickService);
        return "/index.jsp";
    }

    @RequestMapping("/show2")
    public String show2(){
        System.out.println("show2 实现。。。。。"+quickService);
        return "/index.jsp";
    }
}

2.2 请求数据的接收

2.2.1 键值对方式接收数据

接收普通请求数据,当客户端提交的数据是普通键值对形式时,直接使用同名形参接收即可

@Controller
public class PqramController {

    //采用map
    //http://localhost/param4?username=zhangsan&age=12
    @RequestMapping("/param4")
    public String param4(@RequestParam Map<String,String> map){
        map.forEach((k,v)->{
            System.out.println(k+"==>"+  v);
        });
        return "index.jsp";
    }




    //采用集合
    //http://localhost/param2?hobby=swim&hobby=run 路径的接收数据
    @GetMapping("/param3")
    //直接用集合无注解可能会报错,因为接收到了之后会把数据创建完对象,但是接口是不能创建对象的
    //集合用@RequestParam,是告诉springMVC把请求的参数封装进去
    public String param3(@RequestParam List<String> hobby){
        for (String h : hobby) {
            System.out.println(h);
        }

        return "/index.jsp";
    }


    //当要接收同名的多个参数值时,就用数组去接收
    //http://localhost/param2?hobby=swim&hobby=run 路径的接收数据
    @GetMapping("/param2")
    public String param2(String[] hobby){
        for (String h : hobby) {
            System.out.println(h);
        }

        return "/index.jsp";
    }



    //http://localhost/param1? username=zhangsan&age=24 路径的接收数据
    @GetMapping("/param1")
    public String param1(@RequestParam("username") String  name, int age){
        System.out.println(name+"========"+age);
        return "/index.jsp";
    }
}

2.2.1 - I @RquestParam属性
  • @RequestParam可以使获取的数据不创建对象而是直接注入形参,因此使用该注解可以使得参数不一样而注入
    • @RequestParam有三个属性
    • value:要注入参数的名字
    • required:该数据是否必须注入
      • true:是
      • false:否(默认)
    • defaultValue:当数据未提交时的默认值
    //http://localhost/param1?age=24 路径的接收数据
    @GetMapping("/param1")
    public String param1(@RequestParam(value="username",required = true,defaultValue = "magua") String  name, int age){//age的话用int不好,因为如果数据传入没有age就会报错,而用包装类Intage数据没有会返回null
        System.out.println(name+"========"+age);
        return "/index.jsp";
    }

在这里插入图片描述

在这里插入图片描述

2.2.2 封装JavaBean数据

接收实体JavaBean属性数据,单个JavaBean数据:提交的参数名称只要与Java的属性名一致,就可以进行自动封装

username=zhangsan&age=12&hobbyies=swim&hobbies=run&birthday=2019/1/1&address.ciyu=guangdoor&address.area=room
package com.itheima.pojo;

public class Address {
    private String city;
    public String area;

    public String getCity() {
        return city;
    }

    public void setCity(String city) {
        this.city = city;
    }

    public String getArea() {
        return area;
    }

    public void setArea(String area) {
        this.area = area;
    }
}

package com.itheima.pojo;


import java.util.Arrays;
import java.util.Date;

public class User {
    private String username;
    private Integer age;
    private String[] hobbies;
    private Date birthday;
    private Address address;

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    public String[] getHobbies() {
        return hobbies;
    }

    public void setHobbies(String[] hobbies) {
        this.hobbies = hobbies;
    }

    public Date getBirthday() {
        return birthday;
    }

    public void setBirthday(Date birthday) {
        this.birthday = birthday;
    }

    public Address getAddress() {
        return address;
    }

    public void setAddress(Address address) {
        this.address = address;
    }

    public User() {
    }

    @Override
    public String toString() {
        return "User{" +
                "username='" + username + '\'' +
                ", age=" + age +
                ", hobbies=" + Arrays.toString(hobbies) +
                ", birthday=" + birthday +
                ", address=" + address +
                '}';
    }
}

    //http://localhost/param5?username=zhangsan&age=12&hobbyies=swim&hobbies=run&birthday=2019/1/1&address.ciyu=guangdoor&address.area=room
    @RequestMapping("/param5")
    public String param5(User user){
        System.out.println(user);
        return "index.jsp";
    }

在这里插入图片描述

2.2.3 请求参数Postman使用

Postman使用
开发测试时,可以使用Postman进行测试
在这里插入图片描述
@RequestBody:是用于把请求体的这个数据,封装给这个指定的字符串

    @RequestMapping("/param6")
    public String param6(@RequestBody String body ){
        System.out.println(body);
        return "index.jsp";
    }

在这里插入图片描述
在这里插入图片描述

2.2.4 接收JSON格式数据

  • 使用POST请求方式,添加@RequestBody可以使json格式字符串整体打印出来,不会被解析
    例如:
@PostMapping("/Param5")
public String Param5(@RequestBody String body) {
    System.out.println(body);
    return "index.jsp";
}

  • 使用Json工具 ( jackson )将Json格式的字符串转化为JavaBean进行操作
<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>2.9.0</version>
</dependency>
@PostMapping ("/show" )
public string show(@RequestBody string body) throws IOException {
	system.out.println(body);
	//获得objectMapper
	ObjectMapper objectMapper = new ObjectMapper();
	//将json格式字符串转化成指定的User
	User user = objectMapper.readValue(body,User.class);
	System.out.println(user);
	return "/index.jsp";
}

在这里插入图片描述

  • 配置HandlerAdapter可以自动执行转换的工作,指定消息转换器,就不用手动转换json格式字符串了
<!--    配置HandlerMapping-->
    <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"></bean>

<!--    HandlerAdapter-->
    <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter">
        <property name="messageConverters">
            <list>
                <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"></bean>
            </list>
        </property>
@RequestMapping("/param8")
    public String param8(@RequestBody User user ) throws IOException {
        System.out.println(user);
        return "index.jsp";
    }

在这里插入图片描述

2.2.5 接收Restful风格数据

什么是Rest风格?
Rest (Representational State Transfer)表象化状态转变(表述性状态转变),在2000年被提出,基于HTTP、URI
、xml、JSON等标准和协议,支持轻量级、跨平台、跨语言的架构设计。是Web服务的一种新网络应用程序的设计风格和开发方式。

Restful风格的请求,常见的规则有如下三点:
用URI表示某个模块资源,资源名称为名词;

模板URI资源
用户模块userhttp://localhost/user
商品模块product http://localhost/product
账户模块account http://localhost/account
日志模块log http://localhost/log

用请求方式表示模块具体业务动作,例如:GET表示查询、POST表示插入、PUT表示更新、DELETE表示删除

URI资源请求方式参数解释
http://localhost/user/100GET存在URL地址中:100查询id=100的User数据
http://localhost/userPOST存在请求体中Json: {“username”:“haohao” ,“age”:18}插入user数据
http://localhost/userPUT存在请求体中Json : {“id”:100,“username”:“haohao” , “age”:18}修改id=100的User数据
http://localhost/user/100DELETE存在URL地址中:100删除id=100的User数据
http://localhost/product/5GET存在URL地址中:5查询id=5的Product数据
http://localhost/productPOST存在请求体中Json: {“proName”:“小米手机” ,“price”:1299}插入Product数据
http://localhost/productPUT存在请求体中Json : {“id”:5,“proName”:“小米手机” ,“price”:1299}修改id=5的Product数据
http://localhost/product/5DELETE存在URL地址中:5删除id=5的Product数据
  • 用HTTP响应状态码表示结果,国内常用的响应包括三部分:状态码、状态信息、响应数据

    {
    	"code":200,
    	"message":"成功",
    	"data":{
    	"username":"haohao",
    	"age":18
    	}
    }
    {
    	"code":300,
    	"message":"执行错误",
    	"data":"",
    }
    
    

接收Restful风格数据,Restful请求数据一般会在URL地址上携带,可以使用注解@PathVariable(占位符参数
名称)

http://localhost/user/100
@PostMapping ("/user/{id}")
public string findUserById(@Pathvariable("id") Integer id){ 
//@Pathvariable注解表示不采用传统格式解析占位符 
	system.out.println(id);
	return "/index.jsp";
}

请求URL资源地址包含多个参数情况

http://localhost/user/haohao/18
@PostMapping ("/user/{username}/{age}")
public string findUserById(@Pathvariable("username") String username,@Pathvariable("age") Integer age){
	system.out.println(username+"=="+age);
	return "/index.jsp";
}

2.2.6 接收上传的文件

接收文件上传的数据,文件上传的表单需要一定的要求,如下:

  • 表单的提交方式必须是POST
  • 表单的enctype属性必须是multipart/form-data
  • 文件上传项需要有name属性
<form action="" enctype="multipart/form-data" method="post">
	<input type="file" name="myFile">
</form>

http://localhost:8080/SpringMVCDome1/Param6
    @PostMapping("/param10")
    public String Param10(@RequestBody MultipartFile myFile) throws IOException {
        System.out.println(myFile);
        return "/index.jsp";
    }

在这里插入图片描述
在这里插入图片描述

注意:如果上传文件报错This file isn't in your working directory.Teammates you share this requestwith won't be able to use this file.Tomake collaboration easier you cansetup your working directory insettings.

是因为没有修改postman工作区

  • 进入设置file -> settings,上传的文件必须在设置的工作区中
  • 在这里插入图片描述
    在这里插入图片描述
  • 服务器端,由于映射器适配器需要文件上传解析器,而该解析器默认未被注册,所以手动注册

    <!--配置文件上传解析器,注意:id的名字是固定写法-->
    <bean id="multipartResolver"
    class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
    	<property name="defaultEncoding" value="UTF-8"/><!--文件的编码格式默认是Iso8859-1-->
    	<property name="maxUploadSizePerFile" value="1048576"/><!--上传的每个文件限制的大小单位字节-->
    	<property name="maxUploadSize" value="3145728"/><!--上传文件的总大小-->
    	<property name="maxInMemorySize" value="1048576"/><!--上传文件的缓存大小-->
    </bean>
    
  • 而CommonsMultipartResolver底层使用的Apache的是Common-fileuplad等工具API进行的文件上传

    <dependency>
    	<groupId>commons-fileupload</groupId>
    	<artifactId>commons-fileupload</artifactId>
    	<version>1.4</version>
    </dependency>
    

文件上传代码实现

 @PostMapping("/param9")
    public String Param9(@RequestBody MultipartFile myFile) throws IOException {
        System.out.println(myFile);
        //将上传的文件进行保存
        //1、获取当前上传的文件的输入流
        InputStream inputStream = myFile.getInputStream();
        //2、获得上传文件的输出流,被拷贝文件夹的位置
        OutputStream outputStream = new FileOutputStream("E:\\"+myFile.getOriginalFilename());
        //3、执行文件拷贝
        IOUtils.copy(inputStream,outputStream);//(上传的文件,想要被拷贝的文件)
        outputStream.close();
        inputStream.close();
        return "/index.jsp";
    }

复制成功
在这里插入图片描述

2.2.6 获取header头信息和cookie信息

  • 接收Http请求头数据,接收指定名称的请求头

    @GetMapping ("/headers")
    	public String headers(@RequestHeader("Accept-Encoding") String acceptEncoding){
    	System.out.println( "Accept-Encoding:"+acceptEncoding);
    	return "/index.jsp";
    }
    
  • 接收所有的请求头信息

    @GetMapping ("/headersMap")
    public String headersMap(@RequestHeader Map<String,string> map){
    	map.forEach((k, v)->{
    		System.out.println(k+":"+V);
    	});
    	return " /index.jsp" ;
    }
    
  • 获得客户端携带的Cookie数据

    @GetMapping("/cookies")
    public String cookies(@CookieValue(value = "JSESSIONID",defaultValue = "") String jsessionid){
    	System.out.println(jsessionid);
    	return "/index.jsp" ;
    }
    

2.2.7 直接获取Requset和Session域中的数据

获得转发Request域中数据,在进行资源之间转发时,有时需要将一些参数存储到request域中携带给下一个资源

使用HttpServletRequest前导入包

    <dependency>
      <groupId>javax.servlet</groupId>
      <artifactId>javax.servlet-api</artifactId>
      <version>4.0.1</version>
    </dependency>
    @GetMapping("/request1")
    public String request1(HttpServletRequest request) {
        //存储数据
        request.setAttribute("username","haohao");
        return "/request2";
    }

    @GetMapping("/request2")
    public String request2(@RequestAttribute("username") String username) {
        System.out.println(username);
        return "/index.jsp";
    }

在这里插入图片描述
在这里插入图片描述

2.3 Javaweb常用对象获取

直接打印参数request和response,可以发现这两个参数是SpringMVC自动传入的

@GetMapping("/Param10")
public String Param10(HttpServletRequest request, HttpServletResponse response) {
    System.out.println(request);
    System.out.println(response);
    return "/index.jsp";
}

在这里插入图片描述

2.4 请求静态资源

在Spring-MVC框架下,无法直接访问到web文件下的静态资源

静态资源请求的三种解决方案:

  • 第一种方案:在web.xml去配,可以再次激活Tomcat的DefaultServlet,Servlet的url-pattern的匹配优先级是:精确匹配>目录匹配>扩展名匹配>缺省匹配,所以可以指定某个目录下或某个扩展名的资源使用DefaultServlet进行解析。

    <servlet-mapping>
    	<servlet-name>default</servlet-name>
    	<url-pattern>*.html</url-pattern><!--拓展名匹配-->
    </servlet-mapping>
    
    <servlet-mapping>
    	<servlet-name>default</servlet-name>
    	<url-pattern>/img/*</url-pattern><!--目录匹配-->
    </servlet-mapping>
    
  • 第二种方式,在spring-mvc.xml中去配置静态资源映射,匹配映射路径的请求到指定的位置去匹配资源

    <!--mapping是映射资源路径,location是对应资源所在的位置-->
    <mvc:resources mapping="/img/*" location="/img/" />
    <mvc:resources mapping="/css/*" location="/css/" />
    <mvc:resources mapping="/js/*" location="/js/" />
    <mvc:resources mapping="/html/*" location="/html/" />
    

在这里插入图片描述
在这里插入图片描述

  • 第三种方式,在spring-mvc.xml中去配置<mvc:default-servlet-handler >,该方式是注册了一个
    DefaultServletHttpRequestHandler处理器,静态资源的访问都由该处理器去处理,这也是开发中使用最多的
    加了这句话就不用配置路径,比较方便
    <mvc:default-servlet-handler/>
    

2.5 注解驱动< mvc:annotation-driven >标签

  • 静态资源配置的第二第三种方式我们可以正常访问静态资源,但是Controller又无法访问,报错404
    在这里插入图片描述

  • 第二种方式是通过SpringMVC去解析mvc命名空间下的resources标签完成的静态资源解析,第三种方式通过SpringMVC去解析mvc命名空间下的default-servlet-handler标签完成的静态资源解析,根据自定义命名空间的知识,可以发现不管是以上哪种方式,最终都会注册SimpleUrlHandlerMapping

    public BeanDefinition parse(Element elemnt,ParserContext context){
    	//创建SimpleUrlHandlerMapping类型的BeanDefinition
    	RootBeanDefinition handlerMappingDef = new RootBeanDefinition(SimpleUrlHandlerMapping.class);
    	//注册SimpleUrlHandlerMapping的BeanDefinition
    	context.getRegistry().registerBeanDefinition(beanName,handlerMappingDef);
    }
    
  • 又结合组件浅析知识点,一旦SpringMVC容器中存在HandlerMapping类型的组件时,前端控制器
    DispatcherServlet在进行初始化时,就会从容器中获得HandlerMapping,不在加载 dispatcherServlet.properties中默认处理器映射器策略,那也就意味着RequestMappingHandlerMapping不会被加载到了。

  • 手动将RequestMappingHandlerMapping也注册到SpringMVC容器中就可以了,这样DispatcherServlet在进行初始化时,就会从容器中同时获得RequestMappingHandlerMapping存储到DispatcherServlet中名为
    handlerMappings的List集合中,对@RequestMapping注解进行解析。

    <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"/>
    

在这里插入图片描述

该标签内部会帮我们注册RequestMappingHandlerMapping、注册RequestMappingHandlerAdapter并注入Json消息转换器等,上述配置就可以简化成如下:

<!--mvc注解驱动-->
<mvc:annotation-driven/>
<!--配置DefaultServletHttpRequestHandler-->
<mvc:default-servlet-handler/>

PS: <mvcannotation-driven>标签在不同的版本中,帮我们注册的组件不同,Spring 3.0.X版本注册是
DefaultAnnotationHandlerMappingAnnotationMethodHandlerAdapter,由于框架的发展,从Spring 3.1.X开始注册组件变为RequestMappingHandlerMappingRequestMappingHandlerAdapter

3、SpringMVC的响应处理

3.1、传统同步业务数据响应

下面再看一下Spring怎么给客户端响应数据,响应数据主要分为两大部分:

  • 传统同步方式:准备好模型数据,在跳转到执行页面进行展示,此方式使用越来越少了,基于历史原因,一些旧项目还在使用;
  • 前后端分离异步方式:前端使用Ajax技术+Restful风格与服务端进行Json格式为主的数据交互,目前市场上几乎都是此种方式了。

传统同步业务数据响应

  • 请求资源转发
  • 请求资源重定向
  • 响应模型数据
  • 直接回写数据给客户端
    在这里插入图片描述
  • 转发的关键字forword

  • 重定向的关键字redirect

    @GetMapping("/show")
    public String show(){
       System.out.println("show running...."+quickService);
       return "redirect:/index.jsp";
    }
    
    @RequestMapping("/show2")
    public String show2(){
       System.out.println("show2 runnig....."+quickService);
       return "forward:/index.jsp";
    }
    

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

通过视图模型的方式进行转发数据:

@Controller
public class ResponseController1 {

    //重定向方式
    @GetMapping("/res1")
   public String res(){
       System.out.println("res running....");
       return "redirect:/index.jsp";
   }

   //转发方式
   @RequestMapping("/res2")
   public String res2(){
       System.out.println("res2 runnig.....");
       return "forward:/index.jsp";
   }

    @RequestMapping("/res3")
    public ModelAndView res3(ModelAndView modelAndView){
        //ModelAndView用于封装模型数据和视图名

        //设置模型数据
        User user = new User();
        user.setUsername("magua");
        user.setAge(19);
        modelAndView.addObject("user",user);//设置名称,传入user类参数
        //设置视图名称,在页面种展示模型数据
        modelAndView.setViewName("/index.jsp");
        return modelAndView;
    }

}

在这里插入图片描述
直接回写字符串

需要使用Tomcat8

    //直接回写字符串
    @RequestMapping("/res4")
    @ResponseBody//告诉springMVC返回的字符串不是视图,是以响应体方式响应的数据
    public String res4(){
        return "hello magua!";
    }

3.2 前后端分离异步业务数据响应

其实此处的回写数据,跟上面回写数据给客户端的语法方式一样,只不过有如下一些区别:

  • 同步方式回写数据,是将数据响应给浏览器进行页面展示的,而异步方式回写数据一般是回写给Ajax引擎的,即谁访问服务器端,服务器端就将数据响应给谁
  • 同步方式回写的数据,一般就是一些无特定格式的字符串,而异步方式回写的数据大多是Json格式字符串

回写普通数据使用@ResponseBody标注方法,直接返回字符串即可,此处不在说明;
回写Json格式的字符串,即将直接拼接Json格式的字符串或使用工具将JavaBean转换成Json格式的字符串回写

@Controller
@ResponseBody
public class ResponseController2 {

    //以后开发都是用这种方式,直接返回实体
    @GetMapping("/ajax/req3")
//    @ResponseBody
    public User res3() throws JsonProcessingException {
        //设置模型数据
        User user = new User();
        user.setUsername("magua");
        user.setAge(20);

        return user;
    }


    //这种方式也是一个弊端
    @GetMapping("/ajax/req2")
//    @ResponseBody
    public String res2() throws JsonProcessingException {
        //设置模型数据
        User user = new User();
        user.setUsername("magua");
        user.setAge(19);
        //json转换工具
        ObjectMapper mapper = new ObjectMapper();
        String jsonStr = mapper.writeValueAsString(user);

        return jsonStr;
    }
    
    @GetMapping("/ajax/req1")
//    @ResponseBody
    public String res1(){
        return "{\"username\":\"magua\",\"age\":\"18}";
    }
}

在这里插入图片描述

使用@RestController 相当于@ResponseBody和@Controller结合

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

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

相关文章

Linux Hadoop平台伪分布式安装(Hive on Spark)

&#x1f4d4;Linux Hadoop 伪分布式安装(Hive on Spark) 安装目录 1. JDK2. Hadoop3. MysqlHive3.1 Mysql8安装3.2 Hive安装 4. Spark4.1 Maven安装4.2 Scala安装4.3 Spark编译并安装 5. Zookeeper6. HBase 版本概要&#xff1a; jdk&#xff1a; jdk-8u391-linux-x64.tar.gz…

浅析SR隧道路径批量构造方法

为什么要仿真PCE LSP下发隧道路径&#xff1f; 在大型的多区域网络中&#xff0c;路径计算非常复杂。在某些场景下&#xff0c;为了完成路径计算&#xff0c;需要在控制器上部署特殊的计算组件&#xff0c;并需要不同区域中的节点之间协作。这使得网元在进行路径计算时效率低&…

局部路由守卫path守卫

局部路由守卫path守卫 path守卫&#xff08;beforeEnter&#xff09; 代码位置&#xff1a;path守卫代码写在route对象中&#xff08;index.js文件&#xff09;beforeEnter是一个函数&#xff0c;参数上没有回调函数beforeEnter有三个参数&#xff1a; to参数&#xff1a;to是…

CocosCreator | 2.3.3及后续版本浏览器无法断点和控制台不显示错误代码路径的解决方案(cocos代码报错无法定位的问题)

在2.3.3正式版的官方介绍中有这么一项&#xff1a; 提升网页预览时的加载速度 为了进一步提升开发效率&#xff0c;我们优化了网页预览时的脚本加载速度。不论是对引擎还是项目中的代码&#xff0c;载入速度都获得了提升。特别是在开启自定义引擎&#xff0c;或者使用手机扫码…

实时时钟和日历电路MS85163/MS85163M

主要特点 ◼ 基于 32.768kHz 晶振提供年、月、日、 周工作日、小时、分钟和秒 ◼ 具有世纪标记&#xff0c;可工作于 2000-2199 年 ◼ 工作电压&#xff1a; 1.8V-5.5V ◼ 低功耗 ◼ 最高频率达 400kHz 的 I 2 C 接口 ◼ 可编程的时钟输出 (32.768kHz, 1.024kHz…

开发人员请注意:在 PyPI 上的 Python 包中发现 BlazeStealer 恶意软件

1、开发人员请注意&#xff1a;在 PyPI 上的 Python 包中发现 BlazeStealer 恶意软件 一组新的恶意 Python 包已经滑入 Python 包索引 &#xff08;PyPI&#xff09; 存储库&#xff0c;其最终目的是从受感染的开发人员系统中窃取敏感信息。这些软件包伪装成看似无害的混淆工具…

记录一次数据库内存占满

一&#xff0c;背景及表现 用户普遍反馈业务功能&#xff0c;无法加载出数据&#xff0c;服务挂了 二&#xff0c;数据库指标 三&#xff0c;排查方反馈 排查方反馈&#xff1a;查全量数据导致的&#xff0c;就是没查出来但是一直在查 四&#xff0c;分析 1.应用服务器问题…

深度系统v15.4正式发布,惊艳眼球

导读深度操作系统是一个致力于为全球用户提供美观易用、安全可靠的Linux发行版。深度桌面操作系统V15.4专业版使用全新设计的控制中心和重构桌面&#xff0c;采用模糊透明整体风格&#xff0c;全新的热区交互及窗口管理器动效&#xff0c;精挑细选的桌面壁纸&#xff1b;采用全…

VNC连接服务器实现远程桌面 --以AutoDL云服务器为例

VNC连接服务器实现远程桌面 --以AutoDL云服务器为例 针对本地机为Windows 云服务器租显卡跑些小模型很方便&#xff0c;但是当你想做可视化的时候&#xff0c;可能会遇到麻烦&#xff0c;云服务器没有显示输出界面&#xff0c;无法可视化一些检测任务的结果&#xff0c;或者可…

Python爬虫爬取家纺数据并分析

因为时间的原因&#xff0c;没法写一个详细的教程&#xff0c;但是我可以提供一个基本的框架。你需要根据实际情况进行修改和扩展。以下是使用Python的requests库和BeautifulSoup库来爬取网页内容的基本步骤&#xff1a; # 导入所需的库 import requests from bs4 import Beaut…

从零开始搭建SpringCloud微服务集群:Nacos的安装和配置

目录 Nacos集群搭建 1.集群结构图 2.搭建集群 2.1.初始化数据库 2.2.下载nacos 2.3.配置Nacos 2.4.启动 2.5.nginx反向代理 2.6.优化 Nacos集群搭建 1.集群结构图 官方给出的Nacos集群图&#xff1a; 其中包含3个nacos节点&#xff0c;然后一个负载均衡器代理3个Nac…

winui3开发笔记(二)自定义标题栏

参考文章链接&#xff1a;https://www.programminghunter.com/article/46392310600/ 注意事项 获取 AppWindowTitleBar 的实例并设置其颜色属性时&#xff0c;InitializeTitleBar(AppWindow.TitleBar);&#xff0c;只适用于Windows App SDK 1.2及以上&#xff0c;所以如果用w…

基于springboot学生心理咨询评估系统的设计与实现 全套代码 全套文档 附带视频知道教程

springboot学生心理咨询评估系统,springboot vue mysql (毕业论文10784字以上,共30页,程序代码,MySQL数据库) 代码下载: 链接&#xff1a;https://pan.baidu.com/s/1MjiwuWdkVHFQ4toPP1vVrA?pwd4eck 提取码&#xff1a;4eck 【运行环境】 IDEA, JDK1.8, Mysql, Node, Vue …

Unity 跑酷游戏全部脚本(完结)

脚本1 触发器脚本 这个脚本是主角身上的脚本&#xff0c;用于检测是否碰到其他触发器&#xff0c;并做出对应的行为 using System.Collections; using System.Collections.Generic; using UnityEngine; public class ColliidisonTrigger : MonoBehaviour { //触发检测 …

08.Diffusion Model数学原理分析(下)

文章目录 denoising matching term σ t z \sigma_tz σt​z的猜想Diffusion Model for SpeechDiffusion Model for TextMask-Predict 部分截图来自原课程视频《2023李宏毅最新生成式AI教程》&#xff0c;B站自行搜索。 书接上文。 denoising matching term E q ( x t ∣ x 0 …

适用于 iOS 的 10 个最佳数据恢复工具分享

在当今的数字时代&#xff0c;我们的移动设备占据了我们生活的很大一部分。从令人难忘的照片和视频到重要的文档和消息&#xff0c;我们的 iOS 设备存储了大量我们无法承受丢失的数据。然而&#xff0c;事故时有发生&#xff0c;无论是由于软件故障、无意删除&#xff0c;甚至是…

Java 算法篇-深入理解递归(递归实现:青蛙爬楼梯)

&#x1f525;博客主页&#xff1a; 小扳_-CSDN博客 ❤感谢大家点赞&#x1f44d;收藏⭐评论✍ 文章目录 1.0 递归的说明 2.0 用递归来实现相关问题 2.1 递归 - 阶乘 2.2 递归 - 反向打印字符串 2.3 递归 - 二分查找 2.4 递归 - 冒泡排序 2.5 递归 - 冒泡排序2.0 2.6 递归 - 插…

Vite依赖预构建

本文使用的包管理工具是 npm 开发工具是 vscode 本文作为对 vite的了解性内容即可&#xff0c;实际开发中并不会做太多的工作 依赖预构建干了啥 首先vite会找到对应的依赖&#xff0c; 然后调用 esbuild(对js语法进行处理的一个库)&#xff0c; 将其他规范的代码转换成 esmodu…

虚幻C++基础 day3

常见的游戏机制 Actor机关门 创建一个Actor类&#xff0c;添加两个静态网格与一个触发器 UBoxComponentUStaticMeshComponent 头文件&#xff1a; #include “Components/BoxComponent.h”#include “Components/StaticMeshComponent.h” TriggerDoor.h // Fill out your …

Apinto 网关进阶教程,使用 API Mock 生成模拟数据

什么是 API Mock &#xff1f; API Mock 是一种技术&#xff0c;它允许程序员在不依赖后端数据的情况下&#xff0c;模拟 web服务器端 API 的响应。通常使用 API Mock 来测试前端应用程序&#xff0c;而无需等待后端程序构建完成。API Mock 可以模拟任何 HTTP 请求方法&#x…