SpringMVC -- 梗概--贰

1.为什么要配置: mvc:annotation-driven

1>在springMVC的处理流程中,有两个重要组件:HandlerMapping和HandlerAdapter
  分别负责解析Handler和执行Handler
2>如果配置了<mvc:annotation-driven/>,则在项目中自动注册:
  RequestMappingHandlerMapping
  RequestMappingHanderAdapter
  且如上两个组件是对注解开发最新的最全面的支持
3>如果没有配置:<mvc:annotation-driven/>
  则默认使用:
  DefaultAnnotationHandlerMapping
  AnnotationMethodHandlerAdapter两个组件
  而如上两组间已经弃用,且其对注解的支持并不全面,
  比如@ResponseBody是不能被解析的。
*详细的springMVC流程:

 


2.静态资源的访问问题

配置:<mvc:default-servlet-handler/>
1>静态资源:除了Servlet、Controller之外的资源,如:js,css,png,html等
2>当请求静态资源:...xx/xx/xx.js,...xx/xx/xx.css,...xx/xx/xx.png等
 如上请求会逐步回溯到【/】,即会进入DispatcherServlet,则会有HanderMapping
 取查找Hanler,自然无法找到。此时如果没有如上配置,则404.
3>如果有如上配置,则在项目中会自动注册【/**】的一个handler,且此handler
  会在最后映射请求,如果是项目中存在指定的静态资源,则会转向静态资源。

3.RestFul收参(了解) @PathVariable

1>定制方式:
    //如下两个路径都可以访问到如下方法,请求路径不同,则name61和pwd61匹配到的值不同
    //http://localhost:8989/appname/ful/lime/123
    //http://localhost:8989/appname/ful/oracle/456
    //@PathVariable("pwd61")-->获取路径中pwd61部分匹配到的值,并存入对应参数
    @RequestMapping("/ful/{name61}/{pwd61}")
    public String testMVC(@PathVariable("name61")String name,@PathVariable("pwd61")String password){
        System.out.println("name:"+name+" password:"+password);
        return "forward:/index.jsp";
    }
    *注意:所有的/ful/xx/xx的路径都可以访问到如上方法
2>细节:如果路径名 和 参数名 一致,则可以有省略写法,如下
    @RequestMapping("/ful3/a{age}b/h{password}ilo")
    public String testMVC3(@PathVariable Integer age,@PathVariable String password){
        System.out.println("age:"+age+" password:"+password);
        return "forward:/index.jsp";
    }
    *即【@PathVariable Integer age】将路径中名为age的部分匹配到的值存于age中

4.异常管理

4.1 定制异常管理器

public class MyExceptionResolver implements HandlerExceptionResolver{
    /**
     * 主体逻辑:自动捕获Controller中的异常,每当Controller中抛出异常时,就会执行。
     * param:ex=当前抛出的异常
     *       req=请求对象
     *       res=响应对象
     *       handler=抛出异常的控制器方法
     * 返回值:ModelAndView=用来返回错误页面
     */
    public ModelAndView resolveException(HttpServletRequest req,
            HttpServletResponse res, Object handler, Exception ex) {
        ModelAndView mav=new ModelAndView();
        //识别异常
        if(ex instanceof LoginErrorException){
            mav.setViewName("redirect:/error1.jsp");
        }else if(...){}
        ...
        return mav;
    }
}

4.2 声明

<bean class="com.c61.ex.resolver.MyExceptionResolver"></bean>

4.3 异常管理器作用

作用:统一抽取了所有控制器中的异常处理逻辑
抽取前:
    public String login(String username,String password){
        try{
            us.login(username, password);
        }catch(LoginErrorException e){
            return "redirect:/login.jsp";
        }catch(AException e){
            return "redirect:/xxx.jsp";
        }catch(Exception e){

        }
        return "forward:/index.jsp";
    }
抽取后:
    public String login(String username,String password){
        us.login(username,password);
        return "forward:/index.jsp";
    }

5.拦截器:Interceptor,抽取Controller中的冗余功能

5.1 定制

public class MyInterceptor implements HandlerInterceptor{
    /**
     * 在Controller之前执行(常用)
     * 抽取Controller中的冗余功能
     */
    public boolean preHandle(HttpServletRequest req, HttpServletResponse res,
            Object handler) throws Exception {
        //return false;//中断请求,则后续的controller,postHandle...都不再执行
        //return true;//请求继续
        if(xxx){
            String path=req.getContextPath();
            res.sendRedirect(path+"/index.jsp");//中断之前设置错误视图
            return false;
        }
        return true
    }
    /**
     * 在Controller之后,在响应之前,执行(了解)
     * 可以做视图和数据的最终定制
     */
    public void postHandle(HttpServletRequest arg0, HttpServletResponse arg1,
            Object arg2, ModelAndView mav) throws Exception {
    }
    /**
     * 在视图渲染完毕后执行(了解)
     * 资源回收
     */
    public void afterCompletion(HttpServletRequest req,
            HttpServletResponse res, Object handler, Exception ex)
            throws Exception {
    }
}

5.2配置

<mvc:interceptors>
    <mvc:interceptor>
        <!-- 定义要拦截的路径 -->
        <mvc:mapping path="/inter/test"/>
        <mvc:mapping path="/inter/a"/>
        <mvc:mapping path="/inter/b"/>
        <!-- 声明拦截器 -->
        <bean class="com.c61.interceptor.MyInterceptor"/>
    </mvc:interceptor>
    <mvc:interceptor>
        <!-- 定义要拦截的路径 -->
        <mvc:mapping path="/inter/**"/>
        <mvc:exclude-mapping path="/inter/test/**"/>
        <!-- 声明拦截器 -->
        <bean class="com.c61.interceptor.MyInterceptor2"/>
    </mvc:interceptor>
 </mvc:interceptors>
 *注意:
    1>当有多个拦截器同时拦截时,先配置的先拦截。如过Controller1 先后被inter1和inter2拦截,则具体的拦截路程为:
      pre1==pre2==contorller==post2==post1==after2==after1
    2>/* 只能匹配一级路径:/a  /b  /c  /xxxxxx
.     却不能匹配:/a/b  /e/d/c 
      /** 能匹配任意多级路径:/a  /b  /a/b/c  /xxs/xxx/xxx/xx
    3>mvc:exclude-mapping 不能单独使用,必须配合 mvc:mapping 使用
.     作用是在mvc:mapping的基础上排除一些路径

6.上传

6.1 定制上传表单

1>method="post"
2>enctype="multipart/form-data"
<form method="post" enctype="multipart/form-data" action="${pageContext.request.contextPath}/up/test">
    <input type="file" name="file61"/>
    <input type="submit" value="上传"/>
</form>

6.2 定制接收文件的Controller

public String testUP(MultipartFile file61,HttpSession s) throws IllegalStateException, IOException{
    //file61.transferTo(new File("d:/abc/abc.txt"));
    //获取虚拟路径:/up61 对应部署到服务器后的具体的磁盘路径
    String realPath=s.getServletContext().getRealPath("/up61");
    //获取文件名 xxx.txt  xxx.png
    String fileName=file61.getOriginalFilename();
    //拼接写出路径
    String path=realPath+File.separator+fileName;
    //将文件写出到指定目录位置
    file61.transferTo(new File(path));
    return "forward:/index.jsp";
}

6.3 在springMVC容器中注册文件上传解析器

<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
    <!-- 最大允许的上传大小    byte -->
    <property name="MaxUploadSize" value="2097152"></property>
 </bean>
 *ID必须为:multipartResolver

6.4 导包:commons-fileupload-1.3.jar commons-io-2.0.1.jar

6.5 生成唯一的文件名

//32长度 由16进制字符组成的全球唯一的字符串
UUID uuid=UUID.randomUUID();
System.out.println(uuid);
//uuid.toString()可以作为文件名使用,注意要拼接后缀

6.6 回显上传的图片

<!-- target="c61" 将响应展示在名为c61的iframe中(不再用一张新页面承载响应)  -->
<form target="c61" method="post" enctype="multipart/form-data" action="${pageContext.request.contextPath}/up/test">
    <input type="file" name="file61"/>
    <input type="submit" value="上传"/>
</form>
<iframe name="c61" width="200px" height="200px" frameborder="0"></iframe>

7.下载 (了解)

7.1 定制超链接

<a href="${pageContext.request.contextPath}/down/test?name=html教程全.ppt">免费下载</a>

7.2 定制下载的Controller

public String testDown(String name,HttpSession s,HttpServletResponse res) throws FileNotFoundException, IOException{
    String path=s.getServletContext().getRealPath("/up61");
    String filePath=path+File.separator+name;
    //如果文件名中有中文:
    String name2=URLEncoder.encode(name, "utf-8");
    //设置响应头,实现附件形式下载文件(告知浏览器,需要以附件形式接收)
    res.setHeader("content-disposition", "attachment;filename="+name2);
    //输入流读取所有字节,输出流将所有读取到的字节写出
    IOUtils.copy(new FileInputStream(filePath),res.getOutputStream());
    return null;//如上语句,已经响应了请求,则必须return null;
}

8.验证码 见:第三阶段\springMVC\google_验证码\使用.txt

9.与spring集成

1>只是将代码累加在一起而已
2>当SpringMVC的Controller需要Spring的Service时,可以平行获取
3>唯一的注意点:防止容器之间相互污染
  *容器污染防止手段:springMVC扫描控制机器相关组件,spring扫描剩余的
   <context:component-scan base-package="com.c61">...
  *解决方案一:use-default-filters="true"(true=默认值),则默认扫描所有的@Controller,@Service,@Repository
    <context:component-scan base-package="com.c61" use-default-filters="true">
        <!-- 排除某些注解,被排除的注解将不会扫描 -->
        <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
.   </context:component-scan>
. *解决方案二:use-default-filters="false" 不再主动扫描任何注解,在配合上include-filter,实现只扫描某个/些注解
    <context:component-scan base-package="com.c61" use-default-filters="false">
        <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>
posted @ 2016-11-09 14:10  limeOracle  阅读(231)  评论(0编辑  收藏  举报