从0到1学SpringMVC

SpringMVC介绍

SpringMVC是实现了Servlet的Web框架,在Servlet和业务功能之间,充当高级中介的角色,提高了开发效率。

环境搭建 & Hello MVC

目前SpringBoot大行其道,老式的SpringMVC应用,早已销声匿迹。所以,本文的SpringMVC案例,均使用SpringBoot完成。

链接:君子陶陶/learn-spring - Gitee.com

<parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>2.6.7</version>
</parent>

<dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
</dependencies>
@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

@RestController
public class HelloController {

    /**
     * 测试:在浏览器地址栏输入:http://127.0.0.1:8000/hello
     */
    @RequestMapping("/hello")
    public String helloWorld() {
        return "hello world !";
    }
}

绑定请求参数

SpringMVC绑定参数的来源,就是HTTP请求报文。支持多种方式,提取报文里的数据,绑定到方法的参数中。

老式参数绑定

链接:202-mvc-binding-param/src/main/java/com/junzitaotao/mvc/bindingparam/controller/BaseUserController.java · 君子陶陶/learn-spring - Gitee.com

@Controller
@RequestMapping("/jztt/user")
public class BaseUserController {

    /**
     * 测试URL:http://127.0.0.1:8000/jztt/user/query/name?name=jack&id=1
     */
    @RequestMapping(value = "/query/name", method = {RequestMethod.GET})
    public String queryByName(String name, Long id) {
        System.out.println(name);
        System.out.println(id);
        return name;
    }

    @RequestMapping(value = "/create", method = {RequestMethod.POST})
    public String create(UserDTO user) {
        System.out.println(user.toString());
        return user.toString();
    }

    /**
     * 从原始的request对象,开发者手动解析参数
     * <p>
     * 这种方式比较低效,一般不推荐。作为一种兜底方案吧,在高级绑定方式不奏效时,可以使用。
     */
    @RequestMapping("/query/origin")
    public String queryFromRequest(HttpServletRequest request) {
        System.out.println(request.getParameter("name"));
        return request.getParameter("name");
    }
}

注解式参数绑定【推荐】

链接:202-mvc-binding-param/src/main/java/com/junzitaotao/mvc/bindingparam/controller/PopularUserController.java · 君子陶陶/learn-spring - Gitee.com

@Controller
@RequestMapping("/jztt/user")
public class PopularUserController {

    /**
     * 使用@RequestParam注解
     * <p>
     * value表示指定接收的参数名称;
     * required表示是否可以为空,默认true;
     * defaultValue表示为空时的默认值
     */
    @RequestMapping(value = "/query/name/v2", method = {RequestMethod.GET})
    public String queryByNameV2(@RequestParam(value = "name", required = false, defaultValue = "Tom") String name) {
        System.out.println(name);
        return name;
    }

    /**
     * 使用@PathVariable注解
     * <p>
     * 传输参数时可以不通过url?aa=bb&cc=dd的形式,通过url/aa/bb的方式进行参数传输
     * <p>
     * PathVariable是一种REST的API设计风格,设计得好,显得比较优雅。设计功力不够,就有点尴尬了。
     * 另外,解析PathVariable的参数,比较耗性能,高并发场景不推荐。
     */
    @RequestMapping(value = "/query/{name}/v3", method = {RequestMethod.GET})
    public String queryByNameV3(@PathVariable String name) {
        System.out.println(name);
        return name;
    }

    /**
     * 使用@RequestBody注解
     * <p>
     * 将请求体中的数据,反序列化成指定的对象类型,适用于传输数据为一个自定义的对象
     */
    @RequestMapping(value = "/create/v2", method = {RequestMethod.POST})
    public String createV2(@RequestBody UserDTO user) {
        System.out.println(user.toString());
        return user.toString();
    }

    /**
     * 使用@RequestHeader,解析绑定请求头的参数
     */
    @RequestMapping("/query/userAgent")
    public String queryByUserAgent(@RequestHeader("User-Agent") String userAgent) {
        System.out.println(userAgent);
        return userAgent;
    }

    /**
     * 使用@CookieValue获取cookie中的值
     */
    @RequestMapping("/query/cookie")
    public String queryFromCookie(@CookieValue("name") String name) {
        System.out.println(name);
        return name;
    }
}

响应

响应数据【推荐】

前后端分离越来越流行,前端仅依赖后端提供数据,而页面跳转、参数传递等,不再需要SpringMVC做为媒介。所以,后端返回JSON类数据,成为了更流行的开发风格。

链接:君子陶陶/learn-spring - Gitee.com

@Controller
@RequestMapping("/jztt/user")
public class UserController {

    /**
     * 返回普通字符串
     * <p>
     * 测试URL:http://127.0.0.1:8000/jztt/user/query/name?name=jack
     */
    @ResponseBody
    @RequestMapping(value = "/query/name", method = {RequestMethod.GET})
    public String queryByName(String name) {
        System.out.println(name);
        return name;
    }

    /**
     * 返回对象
     * <p>
     * 返回格式:JSON
     */
    @ResponseBody
    @RequestMapping(value = "/query/any", method = {RequestMethod.GET})
    public UserDTO queryAny() {
        UserDTO user = UserDTO.of().setId(123L).setName("Tom").setAge(22);
        System.out.println(user.toString());
        return user;
    }

}

响应页面

跳转页面

异常处理

异常处理,有多种方案。目前最推荐的方式是【@RestControllerAdvice + @ExceptionHandler】的组合模式。

链接:君子陶陶/learn-spring - Gitee.com

/**
 * RestControllerAdvice:对Controller进行增强,返回值为JSON格式
 */
@Slf4j
@RestControllerAdvice
public class ControllerExceptionHandler {
    /**
     * 捕捉所有异常
     */
    @ExceptionHandler(Throwable.class)
    @ResponseStatus(HttpStatus.OK)
    public CommonResult<?> exception(HttpServletRequest request,
                                     HttpServletResponse response,
                                     Exception exception) {
        // 记录异常
        log.warn(exception.getMessage());

        // 返回标准模型
        return CommonResult.ofFail("请求服务异常,请稍后重试");
    }
}

/**
 * 定义通用的返回模型
 */
@Data(staticConstructor = "of")
@Accessors(chain = true)
public class CommonResult<T> implements Serializable {

    private String code;
    private String message;
    private T      data;

    public static CommonResult<?> ofFail(String message) {
        return CommonResult.of().setMessage(message);
    }
}

@Controller
@RequestMapping("/jztt/user")
public class UserController {

    /**
     * 返回对象
     * <p>
     * 返回格式:JSON
     */
    @ResponseBody
    @RequestMapping(value = "/query/any", method = {RequestMethod.GET})
    public UserDTO queryAny() {
        // 模拟异常
        int result = 1 / 0;

        UserDTO user = UserDTO.of().setId(123L).setName("Tom").setAge(22);
        System.out.println(user.toString());
        return user;
    }

}

拦截器

链接:君子陶陶/learn-spring - Gitee.com

public class LogInterceptor implements HandlerInterceptor {

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("============================ preHandle ================================");

        // 返回true,表示不拦截
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("============================ postHandle ================================");
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("============================ afterCompletion ================================");
    }
}

@Configuration
public class UserConfiguration implements WebMvcConfigurer {
    /**
     * 注册拦截器
     * <p>
     * 实现接口WebMvcConfigurer,框架会回调addInterceptors()方法,完成拦截器的注册
     */
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(new LogInterceptor())
                // 拦截所有请求
                .addPathPatterns("/**");
    }
}

@Controller
@RequestMapping("/jztt/user")
public class UserController {

    /**
     * 返回对象
     * <p>
     * 返回格式:JSON
     */
    @ResponseBody
    @RequestMapping(value = "/query/any", method = {RequestMethod.GET})
    public UserDTO queryAny() {
        UserDTO user = UserDTO.of().setId(123L).setName("Tom").setAge(22);
        System.out.println(user.toString());
        return user;
    }
}

特定请求:上传与下载

  • 10
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
本教程是基于MyEclipse环境下基于maven搭建的springmvc步骤(图文教程),文章结尾含有源码下载地址: 新建maven project File->New->Project 选择Maven Project 选择项目路径 Use default Workspace location默认工作空间 选择项目类型 在Artifact Id中选择maven-archetype-webapp 输入Group ID和 Artifact ID,以及Package Group ID一般写大项目名称。Artifact ID是子项目名称。 例如Spring的web包,Group ID:org.springframework,artifactId:spring-web。 Package是默认给你建一个包,不写也可以。 建好后如下图: 接下来需要添加src/main/java,src/test/java ,src/test/resources三个文件夹。右键项目根目录点击New -> Source Folder, 建出这三个文件夹。注意不是建普通的Folder,而是Source Folder。 更改文件夹显示的顺序:右击->属性->Order and Export 更改JDK版本:在Libraries双击JRE System Library,要1.6版本 更改class路径 右键项目,Java Build Path -> Source 下面应该有4个文件夹。src/main/java,src/main/resources,src/test/java ,src/test/resources。   双击每个文件夹的Output folder,选择路径。 src/main/java,src/main/resources,选择target/classes; src/test/java ,src/test/resources, 选择target/test-classes;   选上Allow output folders for source folders. 点击项目名称右击,然后如图选择 在pom.xml中添加所需要的jar包 使用Maven POM editor打开项目中的pom.xml文件,选择Dependencies,在Dependencies栏目点击Add进行,首先弹出一个搜索按钮,例如输入spring-web,就会自动搜索关于spring-web相关的jar包,我们选择3.0.5版本的spring。将spring包全部添加进来。需要添加的其他jar包有:junit、jstl。或者点击pom.xml直接编辑pom.xml文件。这样可以直接copy过来dependencies内容。 设置jar包的scope 当添加进入一个jar包后,有一些属性需要设置,最重要的就是scope,它有以下几种取值: 1.         compile,缺省值,适用于所有阶段,会随着项目一起发布。 2.         provided,类似compile,期望JDK、容器或使用者会提供这个依赖。如servlet.jar。 3.         runtime,只在运行时使用,如JDBC驱动,适用运行和测试阶段。 4.         test,只在测试时使用,用于编译和运行测试代码。不会随项目发布。 5.         system,类似provided,需要显式提供包含依赖的jar,Maven不会在 Repository中查找它。 注意: 在使用maven 打包或者编译时报:-source1.3 中不支持注释错误解决方案如下: <build>   <plugins>    <plugin>     <artifactId>maven-compiler-plugin</artifactId>     <version>2.0.2</version>     <configuration>      <source>1.5</source>      <target>1.5</target>     </configuration>    </plugin>   </plugins>  </build> 添加如下配置就不会再报错了 pom.xml文件代码: <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>wang.gom.mavenspringmvc</groupId> <artifactId>mavenspringmvctest</artifactId> <packaging>war</packaging> <version>0.0.1-SNAPSHOT</version> <name>mavenspringmvctest Maven Webapp</name> <url>http://maven.apache.org</url> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <dependency> <groupId>org.apache.openejb</groupId> <artifactId>javaee-api</artifactId> <version>5.0-1</version> <scope>provided</scope> </dependency> <dependency> <groupId>javax.faces</groupId> <artifactId>jsf-api</artifactId> <version>1.2_04</version> <scope>provided</scope> </dependency> <dependency> <groupId>javax.servlet</groupId> <artifactId>jstl</artifactId> <version>1.2</version> <scope>provided</scope> </dependency> <dependency> <groupId>javax.servlet.jsp</groupId> <artifactId>jsp-api</artifactId> <version>2.1</version> <scope>provided</scope> </dependency> <dependency> <groupId>javax.faces</groupId> <artifactId>jsf-impl</artifactId> <version>1.2_04</version> <scope>provided</scope> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>3.1.2.RELEASE</version> <type>jar</type> <scope>compile</scope> </dependency> <dependency> <groupId>aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.5.4</version> <scope>compile</scope> </dependency> <dependency> <groupId>javax.servlet</groupId> <artifactId>javax.servlet-api</artifactId> <version>3.1.0</version> <type>jar</type> <scope>compile</scope> </dependency> <dependency> <groupId>org.apache.openejb</groupId> <artifactId>commons-dbcp-all</artifactId> <version>1.3-r699049</version> <type>jar</type> <scope>compile</scope> </dependency> <dependency> <groupId>cglib</groupId> <artifactId>cglib</artifactId> <version>2.2.2</version> <type>jar</type> <scope>compile</scope> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>3.1.1.RELEASE</version> <type>jar</type> <scope>compile</scope> </dependency> </dependencies> <build> <finalName>mavenspringmvctest</finalName> <plugins> <plugin> <artifactId>maven-compiler-plugin</artifactId> <version>2.0.2</version> <configuration> <source>1.5</source> <target>1.5</target> </configuration> </plugin> </plugins> </build> </project> web.xml文件代码: <?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" version="2.5" > <!-- 区分项目名称,防止默认重名 --> <context-param> <param-name>webAppRootKey</param-name> <param-value>maven.example.root</param-value> </context-param> <!-- Spring的log4j监听器 --> <listener> <listener-class>org.springframework.web.util.Log4jConfigListener</listener-class> </listener> <!-- 字符集 过滤器 --> <filter> <filter-name>CharacterEncodingFilter</filter-name> <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class> <init-param> <param-name>encoding</param-name> <param-value>UTF-8</param-value> </init-param> <init-param> <param-name>forceEncoding</param-name> <param-value>true</param-value> </init-param> </filter> <filter-mapping> <filter-name>CharacterEncodingFilter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping> <!-- Spring view分发器 --> <servlet> <servlet-name>dispatcher</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>/WEB-INF/dispatcher-servlet.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>dispatcher</servlet-name> <url-pattern>*.do</url-pattern> </servlet-mapping> </web-app> dispatcher-servlet.xml文件代码 <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:context="http://www.springframework.org/schema/context" xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:tx="http://www.springframework.org/schema/tx" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd"> <mvc:annotation-driven /> <context:component-scan base-package="Controller" /> <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"> <property name="prefix" value="/WEB-INF/views/" /> <property name="suffix" value=".jsp" /> </bean> </beans> GeneralController类代码 package Controller; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.RequestMapping; @Controller public class GeneralController { @RequestMapping(value="index.do") public void index_jsp(Model model){ model.addAttribute("ceshi", "测试你好"); System.out.println("index.jsp"); } } Index.jsp页面代码 <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> </head> <body> <c:out value="${ceshi}"></c:out> </body> </html> 运行http://localhost:8080/mavenspringmvctest/index.do结果为:测试你好 源码下载地址:http://www.anqingonline.com/thread-15730-1-1.html

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值