创建和配置Spring MVC框架构建Web应用

1 认识Spring MVC

Spring Web MVC是构建在Servlet API之上的Web框架,自诞生之时就被纳入了Spring框架中。其正式/官方名称为“Spring  Web MVC”,源自其所属的模块(spring-webmvc),但通常被称为“Spring MVC”。

1.1 MVC设计模式(模型(model)-视图(view)-控制器(controller))

是一种软件设计典范,用一种将业务逻辑、数据、界面显示分离的方法组织代码

  • Model(模型):是应用程序中用于处理应用程序数据逻辑的部分,通常模型对象负责在数据库中存取数据。表示数据
  • View(视图):是应用程序中处理数据显示的部分,表示应用程序的用户界面。向控制器提交数据,并显示模型中的数据。显示数据。
  • Controller(控制器):负责接受请求并将其传递给模型以进行适当的操作,并将结果显示在适当的视图中。处理请求

1.2 Spring MVC处理请求使用的组件

  • DispatcherServet(前端控制器):负责接受请求,并根据处理程序映射将请求委托给对应的处理器,也就是控制器。
  • HandlerMapping(处理程序映射):实现将请求和处理器映射,使DispatcherServlet 能将传入的请求转发到适当的控制器。
  • Controller(控制器):作为请求的处理器,提供请求处理逻辑来处理传入的用户请求并生成适当的响应。该响应包含需要在视图上显示的数据。此外,如果需要跳转页面显示相应内容,该响应还会还包含要显示的视图的逻辑名称。
  • ViewResolver(视图解析器): 用于将控制器返回的逻辑视图名称解析为实际的视图。视图解析器负责选择和解析向用户显示的实际视图。
  • View(视图): 用于在浏览器屏幕上向用户显示所需的响应。

1.3 Spring MVC处理请求的过程

2 创建和配置Spring MVC Web应用

配置Spring MVC Web应用的方式有两种:xml配置与Java代码配置。

配置Spring MVC Web应用的过程如下:

  • 创建 Web 应用程序
  • 添加需要的 Spring 框架jar 包和 Spring Mvc框架 jar 包
  • 在web.xml文件中配置 DispatcherServlet
  • 创建控制器
  • 创建视图
  • 创建和配置 Spring MVC 配置文件
  • 部署和运行程序

2.1 xml配置

在src/main/webapp/WEB-INF下创建三个配置文件:web.xml、applicationContext.xml、dispatcher-servlet.xml

2.1.1 web.xml文件配置DispatcherServlet

首先要在web.xml中配置DispatcherServlet,它负责接受请求,并根据处理程序映射将请求委托给对应的处理器,也就是控制器。

<servlet>
	<servlet-name>dispatcher</servlet-name>
	<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
	<load-on-startup>1</load-on-startup>
</servlet>

除了DispatcherServlet外,我们还应在web.xml中配置servlet-mapping。它指定servlet映射,即指定什么请求会被什么servlet处理。

如我们前面所说的,DispatcherServlet,不是真正的用于处理请求的组件,它拦截要处理的请求,根据处理程序映射(HandlerMapping)将请求委托给对应的处理器,也就是控制器。

<servlet-mapping>
	<servlet-name>dispatcher</servlet-name>
	<!-- 这里的/表示所有请求,但是.jsp的请求除外 -->
	<url-pattern>/</url-pattern>
</servlet-mapping>

 在web.xml文件中我们也可以注册监听器,使用的标签是<listener>

<listener>
	<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

ContextLoaderListener监听器的作用是监听程序的启动和关闭,监听到程序启动的时候,会加载spring的配置文件applicationContext.xml,并创建spring根容器。

2.1.1.1 web.xml文件配置spring配置文件位置

使用<context-param>配置spring配置文件的位置,参数名是固定的,叫做contextConfigLocation
spring配置文件可以对应多个,多个可以使用逗号、空格、换行隔开,另外还可以使用*通配符
如果配置文件放到了src下,则需要添加classpath:前缀

<context-param>
	<param-name>contextConfigLocation</param-name>
	<param-value>
		classpath:config/spring-*.xml
	</param-value>
</context-param>
2.1.1.2 web.xml文件配置spring mvc配置文件位置

spring mvc配置文件的默认名称是"servlet的名称-servlet.xml。当然,也可随意命名该文件,spring mvc配置文件中配置的组件主要给DispatcherServlet使用,所以需要通过DispatcherServlet的初始化参数contextConfigLocation来配置spring mvc配置文件的位置。

<servlet>
	<servlet-name>dispatcher</servlet-name>
	<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
	<init-param>
		<param-name>contextConfigLocation</param-name>
		<param-value>classpath:config/spring_mvc.xml</param-value>
	</init-param>
	<load-on-startup>1</load-on-startup>
</servlet>

web.xml文件的完整内容如下:

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns="http://xmlns.jcp.org/xml/ns/javaee"
	xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
	version="3.1">
	
	<!-- spring配置文件默认名称是applicationContext.xml -->
	<!-- 当然,也可以随意命名,如果自己指定了该文件的名称,需要通过上下文参数来配置其位置 -->
	<!-- spring配置文件中的组件是给整个程序用的 -->
	<!-- 参数名是固定的,叫做contextConfigLocation -->
	<!-- spring配置文件可以对应多个,多个可以使用逗号、空格、换行隔开 -->
	<!-- 另外还可以使用*通配符 -->
	<!-- 如果配置文件放到了src下,则需要添加classpath:前缀 -->
	<context-param>
		<param-name>contextConfigLocation</param-name>
		<param-value>
			classpath:config/spring-*.xml
		</param-value>
	</context-param>
	
	<!-- listener用于注册监听器 -->
	<!-- ContextLoaderListener监听器的作用是监听程序的启动和关闭 -->
	<!-- 监听到程序启动的时候,会加载spring的配置文件applicationContext.xml,并创建spring根容器 -->
	<listener>
		<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
	</listener>
	
	<!-- spring mvc配置文件的默认名称是"servlet的名称-servlet.xml" -->
	<!-- 当然,也可随意命名该文件,spring mvc配置文件中配置的组件主要给DispatcherServlet使用 -->
	<!-- 所以需要通过DispatcherServlet的初始化参数contextConfigLocation来配置spring mvc配置文件的位置 -->	
	<servlet>
		<servlet-name>dispatcher</servlet-name>
		<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
		<init-param>
			<param-name>contextConfigLocation</param-name>
			<param-value>classpath:config/spring_mvc.xml</param-value>
		</init-param>
		<load-on-startup>1</load-on-startup>
	</servlet>
	
	<!-- servlet-mapping指定servlet的映射 
		 也就是指定什么样的请求会被该servlet处理
		 DispatcherServlet不是真正用于处理请求的组件,用于拦截要处理的请求
		 做一些事情后,再将请求分发/派发给真正处理请求的控制器
		 DispatcherServlet,我们叫做前端控制器
	-->
	<servlet-mapping>
		<servlet-name>dispatcher</servlet-name>
		<!-- 这里的/表示所有请求,但是.jsp的请求除外 -->
		<url-pattern>/</url-pattern>
	</servlet-mapping>
	
</web-app>

2.1.2 Spring配置文件applicationContext.xml

在这个配置文件中配置Spring,配置全局性的bean,如Service、Dao等组件

<context:component-scan base-package="com.qdu.service" />
 2.1.2.1 Spring配置文件spring-config.xml

applicationContext.xml是spring配置文件的默认名称,一般会将applicationContext,xml的名称改成spring-config.xml。

spring-config.xml的完整内容如下:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns:p="http://www.springframework.org/schema/p"
	xmlns:context="http://www.springframework.org/schema/context"
	xsi:schemaLocation="
	http://www.springframework.org/schema/beans 
	https://www.springframework.org/schema/beans/spring-beans.xsd
	http://www.springframework.org/schema/context 
	http://www.springframework.org/schema/context/spring-context.xsd">

	<!-- 该配置文件是spring配置文件,用于配置全局性的bean,如Service、Dao等组件 -->
	<context:component-scan base-package="com.qdu.service" />

</beans>

注意:如果自己指定了该文件的名称,需要在web.xml中通过上下文参数来配置其位置 

2.1.3 Spring MVC配置文件dispatcher-servlet.xml

在这个文件中配置Spring MVC处理请求时使用的各个组件(1.2)。

1. 配置Controller(控制器):开启对控制器包的扫描,使用@Controller修饰对应的控制器类

<context:component-scan base-package="com.qdu.controller" />

2. 配置ViewResolver(视图解析器):让DispatcherServlet可以将视图名称解析为视图的实际逻辑名称。

<bean
	class="org.springframework.web.servlet.view.InternalResourceViewResolver">
	<property name="prefix" value="/WEB-INF/jsp/" />
	<property name="suffix" value=".jsp" />
</bean>

3. 启用注解驱动的编程模型,这样@RequestMapping等的配置会生效

 mvc:annotation-driven的标记使用会自动注册多个必须的bean,包括RequestMappingHandlerMapping

<mvc:annotation-driven />

4. 配置静态资源

我们配置servlet-mapping时,<url-pattern>写的是/,即拦截所有除了.jsp的请求,所以静态资源也会被拦截,需要针对静态资源单独处理。

可以用default-servlet-handler标记,其表示将静态资源交给web容器的默认servlet来处理

<mvc:default-servlet-handler />

可以使用mvc:resources标记使用单独的处理器处理静态资源

<!-- location:指定静态资源的位置,mapping用于指定访问静态资源的请求url模式 -->
<mvc:resources location="/" mapping="/index.html" />
<mvc:resources location="/css/" mapping="/css/**" />
<mvc:resources location="/js/" mapping="/js/**" />
<mvc:resources location="/img/" mapping="/img/**" />

如果只是希望能够跳转到WEB-INF/jsp目录下的某个jsp页面,并不做其他处理,可直接使用<mvc:view-controller>标记指定请求url和要跳转的视图的名称 

<mvc:view-controller path="/" view-name="index" />
<mvc:view-controller path="/aaa" view-name="index" />
<mvc:view-controller path="/toIndex" view-name="index" />
2.1.3.2 Srping MVC配置文件spring_mvc.xml

一般会将spring mvc配置文件的名称改成spring_mvc.xml。

spring_mvc.xml的完整内容如下:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns:p="http://www.springframework.org/schema/p"
	xmlns:mvc="http://www.springframework.org/schema/mvc"
	xmlns:context="http://www.springframework.org/schema/context"
	xsi:schemaLocation="
	http://www.springframework.org/schema/mvc 
	http://www.springframework.org/schema/mvc/spring-mvc.xsd
	http://www.springframework.org/schema/beans 
	https://www.springframework.org/schema/beans/spring-beans.xsd
	http://www.springframework.org/schema/context 
	http://www.springframework.org/schema/context/spring-context.xsd">

	<!-- 开启对控制器包的扫描,这样@Controller修饰的类会注册为spring管理的bean -->
	<context:component-scan base-package="com.qdu.controller" />

	<!-- 启用注解驱动的编程模型,这样@RequestMapping等的配置会生效 -->
	<!-- mvc:annotation-driven的标记使用会自动注册多个必须的bean,包括RequestMappingHandlerMapping -->
	<mvc:annotation-driven />

	<!-- 配置视图解析器,让DispatcherServlet可以将视图名称解析为视图的实际逻辑名称 -->
	<bean
		class="org.springframework.web.servlet.view.InternalResourceViewResolver">
		<property name="prefix" value="/WEB-INF/jsp/" />
		<property name="suffix" value=".jsp" />
	</bean>
	
	<!-- /拦截的是所有请求,包括静态资源,会导致静态资源的访问报404 -->
	<!-- 需要针对静态资源单独处理 -->
	<!-- 1. default-servlet-handler标记表示将静态资源交给web容器的默认servlet来处理 -->
	<!-- <mvc:default-servlet-handler /> -->
	
	<!-- 2. mvc:resources标记表示使用单独的处理器处理静态资源 -->
	<!-- location:指定静态资源的位置,mapping用于指定访问静态资源的请求url模式 -->
	<mvc:resources location="/" mapping="/index.html" />
	<mvc:resources location="/css/" mapping="/css/**" />
	<mvc:resources location="/js/" mapping="/js/**" />
	<mvc:resources location="/img/" mapping="/img/**" />
	
	<!-- 如果只是希望能够跳转到WEB-INF/jsp目录下的某个jsp页面 -->
	<!-- 并不做其他处理,可直接使用<mvc:view-controller>标记指定请求url和要跳转的视图的名称 -->
	<mvc:view-controller path="/" view-name="index" />
	<mvc:view-controller path="/aaa" view-name="index" />
	<mvc:view-controller path="/toIndex" view-name="index" />
	
</beans>

2.1.4 总结

基于xml配置的项目结构图如下:

 可以发现,关于spring、spring mvc的配置文件统一放到src/main/resources/config下,web.xml放到src/main/webapp/WEB-INF下。

src/main/java/com/qdu主要放controller包和service包。

src/main/webapp下放网络的静态资源(src/main/webapp/static),src/main/webapp/WEB-INF放的除了web.xml还有jsp页面。

2.2 Java代码配置

在src/main/java/com/qdu/config下创建三个配置类MyWebApplicationInitializer.class、SpringConfig.class、SpringMvcConfig.class

2.2.1 MyWebApplicationInitializer类替换web.xml

该类用于替换web.xml文件,用于初始化程序,在程序启动的时候被加载和使用。

如果spring+springmvc程序用的都是java类配置,那么可以直接让该类继承自
AbstractAnnotationConfigDispatcherServletInitializer类,以便简化写法 

让我们回忆一下我们在web.xml中都做了哪些事:

  1. 配置spring配置文件的位置
  2. 配置spring mvc配置文件的位置
  3. 配置DispatcherServlet

那么在该类中我们就应实现这三个功能:

1. 配置spring配置文件的位置:getRootConfigClasses()方法中指定spring配置类有哪些,可以有多个。加载spring配置类形成的是根容器

@Override
protected Class<?>[] getRootConfigClasses() {
	return new Class<?>[] {SpringConfig.class,SpringTx.class};
}

2. 配置spring mvc配置文件的位置:getServletConfigClasses()方法用于指定spring mvc配置类有哪些,可以有多个。加载spring mvc配置类形成的是Servlet上下文容器

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

3. 配置DispatcherServlet:getServletMappings()方法用于指定DispatcherServlet的url映射,也就是DispatcherServlet能拦截的请求的url模式,可以是多个,用一个String[]表示

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

MyWebApplicationInitializer类完整内容如下:

package com.qdu.config;

import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;

//该类用于替换web.xml文件,用于初始化程序,在程序启动的时候被加载和使用
//如果spring+springmvc程序用的都是java类配置,那么可以直接让该类继承自
//AbstractAnnotationConfigDispatcherServletInitializer类,以便简化写法
public class MyWebApplicationInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {

	//getRootConfigClasses()方法中指定spring配置类有哪些,可以有多个
	//加载spring配置类形成的是根容器
	@Override
	protected Class<?>[] getRootConfigClasses() {
		return new Class<?>[] {SpringConfig.class,SpringTx.class};
	}

	//getServletConfigClasses()方法用于指定spring mvc配置类有哪些,可以有多个
	//加载spring mvc配置类形成的是Servlet上下文容器
	@Override
	protected Class<?>[] getServletConfigClasses() {
		return new Class<?>[] {SpringMvcConfig.class};
	}

	// getServletMappings()方法用于指定DispatcherServlet的url映射
	// 也就是DispatcherServlet能拦截的请求的url模式,可以是多个,用一个String[]表示
	@Override
	protected String[] getServletMappings() {
		return new String[] { "/" };
	}
}

2.2.2 SpringConfig类替换spring-config.xml

该类用于替换spring-config.xml文件

同spring-config.xml中所作的事情一样,开启对com.qdu.service(还可以有dao)的包扫描

package com.qdu.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;

@Configuration //说明该类是个配置类
@ComponentScan(basePackages= {"com.qdu.service"})
public class SpringConfig {

}

2.2.3 SpringMVCConfig类替换spring_mvc.xml

该类用于替换spring mvc的配置xml文件,Spring MVC的配置类需要实现WebMvcConfigurer接口,来实现各种mvc配置。

我们在spring_mvc.xml中共做了如下几件事:

  1. 配置控制器Controller
  2. 配置视图解析器ViewResolver
  3. 启用注解驱动的编程模型,这样@RequestMapping等的配置会生效
  4. 配置静态资源访问

1. 对于Controller,使用@ComponentScan开启对Controller包的扫描

@ComponentScan(basePackages = { "com.qdu.controller" })

2. 使用configureViewResolvers()方法中注册视图解析器组件

@Override
public void configureViewResolvers(ViewResolverRegistry registry) {
	//调用jsp()方法指定前缀和后缀
	registry.jsp("/WEB-INF/jsp/", ".jsp");
}

3. @EnableWebMvc和在xml配置文件中使用<mvc:annotation-driven />等效

@EnableWebMvc

4. 使用addResourceHandlers()中添加对静态资源的处理,为指定的静态资源添加资源处理器,让静态资源能够正常显示。
 /static/**是访问静态资源的url模式,/static/是静态资源所在的位置

@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
	// /static/**是访问静态资源的url模式,/static/是静态资源所在的位置
	registry.addResourceHandler("/static/**").addResourceLocations("/static/");
}

addViewControllers()方法用于为一些请求添加简单的视图控制器,这样可以不用添加控制器方法实现简单的页面跳转

@Override
public void addViewControllers(ViewControllerRegistry registry) {
	registry.addViewController("/").setViewName("index");
	registry.addViewController("/aaa").setViewName("index");
	registry.addViewController("/toIndex").setViewName("index");
	registry.addViewController("/toSecond").setViewName("second");
}

在这里详细解释一下这里的“添加控制器方法实现简单的页面跳转”:

WEB-INF/jsp目录下的页面是不能直接访问,需要经过控制器跳转,比如我们现在有一个index.jsp,我们要跳转到这个页面需要在com.qdu.controller包中写一个IndexController

package com.qdu.controller;

import org.springframework.stereotype.Controller;

@Controller
public class IndexController {

	@RequestMapping({"/","/aaa","/toIndex"})
	public String toIndex() {
		return "index"; //返回的index是要跳转的页面的名称
	}
	
}

这里的"/"这个url表示程序的上下文路径,意思是如果请求url是上下文路径,就跳转到index页面/视图,可以对应多个url。

如果需要添加方法处理请求,比如需要从数据库读取一些数据,再跳转到页面显示,可以在控制类中添加方法处理请求。
但是如果只是单纯希望跳转到一个jsp页面,不做其他处理,直接在springmvc配置文件中配置即可

 SpringMVCConfig类的内容如下:

package com.qdu.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.EnableWebMvc;
import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
import org.springframework.web.servlet.config.annotation.ViewControllerRegistry;
import org.springframework.web.servlet.config.annotation.ViewResolverRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

@Configuration
@ComponentScan(basePackages = { "com.qdu.controller" })
//Spring MVC的配置类需要实现WebMvcConfigurer接口,来实现各种mvc配置
//@EnableWebMvc和在xml配置文件中使用<mvc:annotation-driven />等效
@EnableWebMvc 
public class SpringMvcConfig implements WebMvcConfigurer {
	//addResourceHandlers()中添加对静态资源的处理
	@Override
	public void addResourceHandlers(ResourceHandlerRegistry registry) {
		//  为指定的静态资源添加资源处理器,让静态资源能够正常显示
		// /static/**是访问静态资源的url模式,/static/是静态资源所在的位置
		registry.addResourceHandler("/static/**").addResourceLocations("/static/");
	}
	//addViewControllers()方法用于为一些请求添加简单的视图控制器
	//这样可以不用添加控制器方法实现简单的页面跳转
	@Override
	public void addViewControllers(ViewControllerRegistry registry) {
		registry.addViewController("/").setViewName("index");
		registry.addViewController("/aaa").setViewName("index");
		registry.addViewController("/toIndex").setViewName("index");
		registry.addViewController("/toSecond").setViewName("second");
	}
	//configureViewResolvers()方法中注册试图解析器组件
	@Override
	public void configureViewResolvers(ViewResolverRegistry registry) {
		//调用jsp()方法指定前缀和后缀
		registry.jsp("/WEB-INF/jsp/", ".jsp");
	}
}

相关推荐

  1. Nodejs--构建web应用

    2023-12-28 12:30:05       7 阅读
  2. 使用 Express.js MySQL 构建 Web 应用程序

    2023-12-28 12:30:05       33 阅读
  3. web项目创建流程框架

    2023-12-28 12:30:05       42 阅读
  4. Web框架开发-Django-模板继承静态文件配置

    2023-12-28 12:30:05       20 阅读

最近更新

  1. TCP协议是安全的吗?

    2023-12-28 12:30:05       18 阅读
  2. 阿里云服务器执行yum,一直下载docker-ce-stable失败

    2023-12-28 12:30:05       19 阅读
  3. 【Python教程】压缩PDF文件大小

    2023-12-28 12:30:05       18 阅读
  4. 通过文章id递归查询所有评论(xml)

    2023-12-28 12:30:05       20 阅读

热门阅读

  1. 换页的算法以及例题

    2023-12-28 12:30:05       38 阅读
  2. 在Visual Studio(VS)编译器中,Release和Debug区别

    2023-12-28 12:30:05       37 阅读
  3. 【Web2D/3D】SVG(第二篇)

    2023-12-28 12:30:05       41 阅读
  4. *CSS: 级联样式表

    2023-12-28 12:30:05       35 阅读
  5. lodash源码分析每日一练 - 数组 - join

    2023-12-28 12:30:05       33 阅读
  6. Hive的视图和索引

    2023-12-28 12:30:05       41 阅读
  7. 31--设计模式、面向对象设计原则

    2023-12-28 12:30:05       31 阅读
  8. 探索Go语言的魅力:一门简洁高效的编程语言

    2023-12-28 12:30:05       32 阅读
  9. Mybatis 动态 SQL - foreach

    2023-12-28 12:30:05       40 阅读
  10. oj 1.8编程基础之多维数组 16:矩阵剪刀石头布

    2023-12-28 12:30:05       35 阅读