【玩转Spring】spring mvc

M:Model 模型

V:View 视图

C:Controller 控制器

Spring MVC是基于模型-视图-控制器模式实现的。不管你是struts,还是Spring MVC,只要是基于Java的WEB框架,都会通过一个前端控制器器。在Spring MVC中DispatcherServlet就是它的前端控制器,那么这个前端控制器做了什么呢?

DispatcherServlet


明白了DispatcherServlet是Spring MVC的前端控制器,那么我们又是怎么将请求全部先发给前端控制器,然后由前端控制器来控制跳转到相应的组件呢?

注意,如果要使用注解的方式启动MVC,你的项目必须部署在支持servlet3.0的容器当中,如tomcat7或者更好的版本;这是由于在Servlet 3.0环境中, 容器会在类路径中查找实现javax.servlet.ServletContainerInitializer接口的类, 如果能发现的话, 就会用它来配置Servlet容器。而Spring提供了这个接口的实现, 名为SpringServletContainerInitializer, 这个类反过来又会查找实现WebApplicationInitializer的类并将配置的任务交给它们来完成。

程序一:配置前端控制器

/***********************DispatcherServlet*******************/
public class ServletInit extends AbstractAnnotationConfigDispatcherServletInitializer{
    public ServletInit() {
        System.out.println("dispatcherservlet启动了");
    }

    //指定非WEB相关的配置类
    @Override
    protected Class<?>[] getRootConfigClasses() {
        return new Class[] {RootConfig.class};
    }

    //指定WEB启动的配置类
    @Override
    protected Class<?>[] getServletConfigClasses() {
        return new Class[] {WebConfig.class};
    }

     //将DispatcherServlet映射到 "/"
    @Override
    protected String[] getServletMappings() {
        return  new String[] {"/"};
    }
}

/*************************RooConfig**********************/
@Configuration
@ComponentScan(basePackages= {"mvc.logic"})
public class RootConfig {
    public RootConfig() {
        System.out.println("RootConfig启动");
    }
}

/*************************WebConfig**********************/
@Configuration
@ComponentScan(basePackages= {"mvc.web"})
@EnableWebMvc
public class WebConfig {
    public WebConfig() {
        System.out.println("WebConfig启动");
    }
}

上面的代码有两个问题

1、没有配置视图解析器

2、对静态资源也进行了拦截

视图解析器

视图解析器的作用是将逻辑视图转为物理视图,所有的视图解析器都必须实现ViewResolver接口。Spring MVC将按照你配置的不同的视图解析器来对模板进行渲染。Spring为提供了对多种视图的支持,常用的是以下三种:

  • InternalResourceViewResolver 将视图解析为JSP

  • FreeMarketViewResolver 将视图解析为FreeMarker模板

  • ThymeleafViewResolver 将视图解析为Thymeleaf模板

程序二:JSP视图解析器

@Configuration
@ComponentScan(basePackages= {"mvc.web"})
@EnableWebMvc
public class WebConfig extends WebMvcConfigurerAdapter{
    public WebConfig() {
        System.out.println("WebConfig启动");
    }

    //配置JSP视图解析器
    @Bean
    public ViewResolver viewResolver() {
        InternalResourceViewResolver  viewreResolver = new InternalResourceViewResolver("/WEB-INF/views/",".jsp");
        return viewreResolver;
    }

    //配置静态资源过滤
    @Override
    public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
        configurer.enable();
    }

}

控制器

控制器是真正对数据进行处理的地方,主要包括了参数的接收和view的返回。

程序三:简单的控制器

@Controller
public class UserController {
    public UserController() {
        System.out.println("UserController被启动了");
    }

    /*************表示接收处理路径为index的,GET方式的请求********/
    @RequestMapping(value="/index",method=RequestMethod.GET)
    public void index() {
        System.out.println("调用了index");
    }
}

控制器其实就是一个类,只不过使用@RequestMaaping将方法与路径进行了绑定。

参数的接收

@ReqeustParame == request.getParameter

@RequestMapping(value="/index/",method=RequestMethod.GET)
public void index(@RequestParam("userName") String userName ) {
    System.out.println("调用了index");
}

@PathVariable

@RequestMapping(value="/index/{userName}",method=RequestMethod.GET)
public void index(@PathVariable("userName") String userName ) {
    System.out.println("调用了index");
}

视图返回

返回一个视图名

@RequestMapping(value="/index/",method=RequestMethod.GET)
public String index(@RequestParam("userName") String userName ) {
    return "index";
}

返回ModelAndView

@RequestMapping(value="/index/",method=RequestMethod.GET)
public ModelAndView index(@RequestParam("userName") String userName ) {
    ModelAndView modelAndView = new ModelAndView();
    modelAndView.addObject("", "");
    modelAndView.setViewName("");
    return modelAndView;
}

文件上传

首先我们想一想文件上传我们在服务端知道些什么?

  • 文件名

  • 文件类型

  • 文件大小

  • 文件

那么Spring MVC是怎么处理这些问题的呢?

上一节我们已经实现自定义DispatcherServlet,但是在DispatcherServlet中并未实现任何解析multipart请求数据的功能。它将该任务委托给了Spring中MultipartResolver策略接口的实现, 通过这个实现类来解析multipart请求中的内容。 从Spring 3.1开始, Spring内置了两个MultipartResolver的实现供我们选择:CommonsMultipartResolver( 使用Jakarta Commons FileUpload解析multipart请求);StandardServletMultipartResolver(依赖于Servlet 3.0对multipart请求的支持)。一般来讲, 在这两者之间, StandardServletMultipartResolver可能会是优选的方案(不依赖于外部组件)。

程序一:在WebConfig中配置MultipartResolver

@Bean
public MultipartResolver multipartResolver() {
    return new StandardServletMultipartResolver();
}

同时在DispatcherServlet中,你还需要重写customizeRegistration函数。

程序二:配置文件上传的相关参数

@Override
protected void customizeRegistration(Dynamic registration) {
    String location = "E:/spring-mvc/tmp/uploads";
    File file = new File(location);
    if(!file.exists()) {
        file.mkdirs();
    }
        //每一个文件为3M
    long maxFileSize = 1024 * 1024 * 3;
        //一共上传15M的内容 
    long requestFileSzie = maxFileSize * 5; 
        //当缓存中有好大的时候,写入磁盘
    int fileSizeThreshold = 0; 
    registration.setMultipartConfig(
                new MultipartConfigElement(
                        location, 
                        maxFileSize,
                        requestFileSzie ,
                        fileSizeThreshold));
}

程序三:在controller中获取MultipartFile数据

@RequestMapping(value="/upload")
public String upload(@RequestPart("myFile") MultipartFile myFile) {
    System.out.println("文件名称:"+myFile.getOriginalFilename());
    return "";
}

至此,Spring MVC文件上传就完了,是不是非常简单,非常感谢Spring为我们带来如此简便的文件上传方法。

异常处理

在Http中,大家经常会碰到404、500等常见异常错误码,但是我们不可以直接将错误码返回给用户,那么我们应该怎么做?在前面的Spring AOP中讲过,可以将所有的异常进行统一处理,但是又怎么返回到指定界面呢?

Spring提供了多种方式将异常转换为响应:

  • 特定的Spring异常将会自动映射为指定的HTTP状态码;

  • 异常上可以添加@ResponseStatus注解, 从而将其映射为某一个HTTP的状态码

  • 在方法上可以添加@ExceptionHandler注解, 使其用来处理异常。

第一种和第二种方式是指将特定情况下的异常转换为HTTP状态码。第三种是对异常的处理。

程序四:@ResponseStatus

@ResponseStatus(value=HttpStatus.INTERNAL_SERVER_ERROR)
public class MyExcetion extends RuntimeException{
}

需要注意的的是,@ResponseStatus是注解在异常类上的

当得到我们需要的异常之后,我们需要对异常进行处理,Spring MVC利用了AOP的原理,加入了@ControllerAdvice注解,此注解能够拦截所有我们定义的异常

程序五:@ControllerAdvice + @ExceptionHandler

@ControllerAdvice
public class ExceptionAdvice {
    @ExceptionHandler(MyExcetion.class)
    public String exception() {
        return "error/500";
    }
}

Spring中的过滤器

程序一:自定义Filter

/////实现Filter类,自定义Filter
public class SessionFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        System.out.println("过滤器启动");
    }
    @Override
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
            throws IOException, ServletException {
        System.out.println("拦截到了访问请求 ");
    }
    @Override
    public void destroy() {
        System.out.println("过滤器销毁");
    }
}

/////web.xml中定义Filter
<filter>
   <filter-name>sessionFilter</filter-name>
   <filter-class>myfilter.SessionFilter</filter-class>
</filter>
<filter-mapping>
   <filter-name>sessionFilter</filter-name>
   <url-pattern>*</url-pattern>
</filter-mapping>

1、Filter在启动时会被初始化,调用一次init方法,且只会初始化一次
2、按照XML定义顺序进行拦截

Spring 拦截器

spring拦截器要实现的功能从名称就看出,那就是拦截用户的请求,功能相似于过滤器。那么它们有什么不同呢?

不管怎么说,把拦截器先运行起来。在Webconfig配置文件中提供了一个addInterceptors函数来完成注册自定义拦截器,就这么简单任性。

程序二:自定义拦截器

/////自定义Interceptor
@Component
public class SessionInterceptor implements HandlerInterceptor{

    public SessionInterceptor() {
        System.out.println("---SessionInterceptor---");
    }

    public boolean preHandle(HttpServletRequest request, 
HttpServletResponse response, Object handler)
            throws Exception {
        System.out.println("---preHandle----");
        return true;
    }

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

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

}

/////注册sessionInterceptor
@Override
public void addInterceptors(InterceptorRegistry registry) {
    registry.addInterceptor(sessionInterceptor).addPathPatterns("/user/*");
}

如果你以为Spring拦截器是仿照Filter来拦截URL那说明你太简单了。Spring拦截器其实是利用了Aop的原理。正是因为如此,我们才能看到上面的preHander、postHander、afterCompltion。

  • preHander:被@RequestMapping注解的方法执行前调用

  • postHander:被@RequestMapping注解的方法执行后未返回ModelView之前调用

  • afterCompltion:方法执行完成后调用

preHahder如果返回false,则postHander不执行。
多个拦截器的执行顺序与注册顺序相关

现在我们再来看Spring MVC的调用顺序,就一目了然了。先通过自定义DispatcherServlet注解启动配置类的方式启动Spring + MVC。实际真正起分发作用的还是org.springframework.web.servlet.DispatcherServlet.doServiet()方法。


知识星球
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容