我们程序中的上下文对象是通过new ClasspathXmlApplicationContext(spring配置文件) 方式获取的,但是每次从容器中获得Bean时都要编写new ClasspathXmlApplicationContext(spring配置文件),这样的弊端是配置文件加载多次,应用上下文创建多次。
在web项目中,可以使用ServletContextListener监听web应用的启动,在应用启动的时候,我们就加载spring的配置文件,创建应用上下文对象ApplicationContext,再将其存储到最大的域servletContext中,这样就可以在任意位置从域中获取应用上下文ApplicationContext对象了。
Spring集成web组件
对于上述的分析Spring框架已经帮通过ContextLoaderListener监听器进行了封装,在该监听器内部加载Spring配置文件,创建上下文对象,并存储到ServletContext域中,提供了一个客户端工具WebApplicationContextUtils供使用者获取上下文对象。
要使用这个东西,我们首先需要在web.xml文件中配置ContextLoaderListener监听器(导入spring-web坐标)。
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>5.0.5.RELEASE</version>
</dependency>
之后需要我们在web.xml中配置contextConfigLocation监听器
首先这个监听器需要一个全局参数,在这个全局参数我们需要告诉监听器,你在web应用创建的加载哪个文件。
<!--全局参数-->
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
之后我们需要配置监听器,将监听器激活一下
<!--Spring的监听器-->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
到此我们就将通过监听器的方式,在项目加载的时候将配置文件存到serviceContext域中了。这里只是在spring框架中集成的web环境,与SpringMVC还没有任何关系。
SpringMVC是一个表现层(UI)框架,感觉无论是.net framework还是SpringMVC都只是表现层的一个框架,与MVC的理念有点区别,但是实际使用MVC就只是一个表现层框架。
SpringMVC可以通过一个简单的注解,将java类变成一个处理请求的控制器,而且不需要实现接口,虽然说很不错,但是我个人观点跟.net的MVC框架集成相比有点差距,可能是.net的更好理解吧,给我的感觉是.net更符合MVC的设计。
MVC 简单理解
MVC简单来讲就是用户请求过来之后,通过一个控制器返回一个视图信息,再去解析这个视图展示。可能不同的语言或者服务器有不同的实现流程。
举个例子:.net里面的IIS服务器,通过管道的方式将请求流到了handler里面,找到对应的处理操作时候返回界面。(不太清楚这里的流程对不对,现在使用IIS的好像不太多)
而SpringMVC是依托Spring框架实现的,它的流程是请求到了前端控制器(DispatcherServlet)中,前端控制器收到请求之后调用处理器映射器(HandlerMapping),处理器映射器找到相应的处理器和处理器拦截器等,将它们封装成一个处理器的执行链,返回给DispatcherServlet(前端控制器),之后,DispatcherServlet去根据这个执行链去寻找处理器适配器(HandlerAdapter),再由HandlerAdapter(处理器适配器)来调用相应的控制器(controller),controller执行完之后,返回一个ModeAndView,再由HandlerAdapter(处理器适配器)返回给前端控制器(DispatcherServlet),然后DispatcherServlet将其传给视图解析器(ViewReslover),解析完成之后返回具体的view。DispatcherServlet根据视图和模型渲染,之后给用户响应。
SpringMVC配置流程
既然SpringMVC只是一个web的表现层框架,所以在使用之前需要在项目中集成web环境哦~
集成好之后,开始第一步:导坐标
导入Spring,SpringMVC,Servlet,Jsp
<!--Spring坐标-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.5.RELEASE</version>
</dependency>
<!--SpringMVC坐标-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.0.5.RELEASE</version>
</dependency>
<!--Servlet坐标-->
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
<!--Jsp坐标-->
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.0</version>
</dependency>
在web.xml中配置SpringMVC的核心控制器
<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>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>DispatcherServlet</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
<init-param>是把我们的springMVC的配置文件加载进来,<load-on-startup>1</load-on-startup>表示项目一启动就会加载这个前端控制器
接下来就是创建控制器,页面
之后是配置注解
@Controller
public class QuickController {
@RequestMapping("/quick")
public String quickMethod(){
System.out.println("quickMethod running.....");
return "index";
}
}
其中RequestMapping注解是配置这个控制器的访问路径的。
配置spring-mvc.xml
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
<!--配置注解扫描-->
<context:component-scan base-package="com.itheima"/> </beans>
这里的重点是配置注解扫描。
首先解析一下RequestMapping
用于建立请求 URL 和处理请求方法之间的对应关系
value:用于指定请求的URL。它和path属性的作用是一样的
method:用于指定请求的方式
params:用于指定限制请求参数的条件。它支持简单的表达式。要求请求参数的key和value必须和配置的一模一 样
params = {"accountName"},表示请求参数必须有accountName
SpringMVC有默认组件配置,默认组件都是DispatcherServlet.properties配置文件中配置的,该配置文件地址 org/springframework/web/servlet/DispatcherServlet.properties,该文件中配置了默认的视图解析器
org.springframework.web.servlet.ViewResolver=org.springframework.web.servlet.view.Inter nalResourceViewResolver
翻看该解析器源码,可以看到该解析器的默认设置,如下:
REDIRECT_URL_PREFIX = "redirect:" --重定向前缀
FORWARD_URL_PREFIX = "forward:" --转发前缀(默认值)
prefix = ""; --视图名称前缀
suffix = ""; --视图名称后缀
我们可以通过属性注入的方式修改视图的的前后缀:
<!--配置内部资源视图解析器-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/views/"></property>
<property name="suffix" value=".jsp"></property>
</bean>
SpringMVC的请求与响应:
在controller中默认是跳转页面,如果不想跳转页面可以使用注解:@ResponseBody标注返回的不是页面,而是其他的字符格式的数据,但是需要我们在spring-web.xm文件中配置,SpringMVC为了简化我们的开发,我们在spring-web.xml文件中可以使用 <mvc:annotation-driven />配置自动加载处理映射器和处理适配器。默认底层会集成jackson进行对象或集合的json格式字符串的转换。