springmvc

一、什么是spring MVC

  spring MVC是一个基于MVC模式的一个轻量级web框架,它本身就是spring框架的一个模块,所以可以和spring框架无缝结合。其可以简化web开发,是一个比struts2更加方便的MVC框架。

二、springMVC的执行流程  

  1.DispatcherServlet :前端控制器,DispatcherServlet拦截请求对URL进行解析,得到请求资源标识符(URI)。然后根据该URI,调用HandlerMapping;

  2.HandlerMapping : 处理器映射,DispatcherServlet调用该处理映射器,获得该Handler配置的所有相关的对象(包括Handler对象以及Handler对象对应的拦截器),最后以HandlerExecutionChain对象的形式返回给DispatcherServlet ;

  3.HandlerAdapter :处理器适配器,DispatcherServlet或得了Handler对象之后,就会调用对应的HandlerAdapter来对请求进行处理(执行对应的拦截器和Handler),并返回ModelAndView给DispatcherServlet。

  4.Handler :处理器,HandlerAdapter执行Handler的方法并返回ModelAndView给HandlerAdapter。

  5.ViewResolver:视图解析器,DispatcherServlet得到了ModelAndView后,调用ViewResolver来解析得到view(真正的视图对象)。

  6.DispatcherServlet 有了view之后,就可以利用Model中的数据对视图进行渲染,并响应用户的请求。

  流程图大致如下:

     

三、springMVC的使用

  1.引入jar包,引入springMVC相关jar包,文件上传jar包,json转换相关jar包

    

  2.在web.xml中配置springMVC核心过滤器

  <!-- 配置spring MVC核心过滤器 -->
  <servlet>
    <servlet-name>dispatcherServlet</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <init-param>
        <param-name>contextConfigLocation</param-name>
        <!-- 该配置文件默认是WEB-INF目录下上面的servlet-name-servlet.xml,如:dispatcherServlet-servlet.xml -->
        <param-value>classpath*:spring-mvc.xml</param-value>
    </init-param>
    <load-on-startup>0</load-on-startup>
  </servlet>
  <servlet-mapping>
    <servlet-name>dispatcherServlet</servlet-name>
    <url-pattern>/</url-pattern>
  </servlet-mapping>
  
  <!-- 解决中文乱码问题(无法解决get请求的乱码问题) -->
  <filter>
    <filter-name>encodingFilter</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>encodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>
  
  <!-- contextLoaderListener监听器的作用就是启动Web容器时,自动装配ApplicationContext的配置信息 -->
  <!-- 如果只用springMVC这个配置可以不要 -->
  <listener>
     <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
  <context-param>  
  <param-name>contextConfigLocation</param-name>  
     <param-value>classpath:applicationContext.xml</param-value>  
  </context-param> 

   3.总共有三种方式来创建处理器Handler,

    ①实现Controller接口,并实现其handleRequest方法,这种方式一个实现类中只有一个方法会被请求到。

      ②继承MultiActionController类,自己新建方法来进行映射,这种方式一个类中可以有多个方法来响应请求。

    ③用注解的方式声明,@Controller或@RestController

    

public class HelloWorldController implements Controller {

    @Override
    public ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception {
        
        System.out.println("传入的数据为");
        String userName=request.getParameter("userName");
        System.out.println("userName:"+userName);
        
        //封装数据,可以直接使用request对象,也可以使用封装等方式,真正使用时可以选择一种
        request.setAttribute("rUserName", userName);
        Map<String, String > map=new HashMap<String, String>();
        map.put("mUserName",userName);
        
        //返回视图层,如果使用map封装数据,需要作为(第二个)参数传递,也是request作用域。
        return new ModelAndView("forward:/index.jsp",map);
    }
}


public class HelloSpringMvcController extends MultiActionController {

    public ModelAndView add(HttpServletRequest request,HttpServletResponse response){
        String userName=request.getParameter("userName");
        System.out.println("传入的数据为userName:"+userName);
        
        //封装数据,可以直接使用request对象,也可以使用封装等方式,真正使用时可以选择一种
        request.setAttribute("rUserName", userName);
        Map<String, String > map=new HashMap<String, String>();
        map.put("mUserName",userName);
        
        //返回视图层,如果使用map封装数据,需要作为(第二个)参数传递,也是request作用域。
        return new ModelAndView("/index.jsp",map);
    }
    
    public ModelAndView update(HttpServletRequest request,HttpServletResponse response){
        String userName=request.getParameter("userName");
        System.out.println("传入的数据为userName:"+userName);
        
        //封装数据,可以直接使用request对象,也可以使用封装等方式,真正使用时可以选择一种
        request.setAttribute("rUserName", userName);
        return new ModelAndView("/index.jsp");
    }
}

@Controller
@RequestMapping("/annotion")
public class AnnotionController{

    /**
     * 下面5个参数是springMVC默认支持的参数类型,我们只需在方法上给出这些参数就能够使用
     * HttpServletRequest request
     * HttpServletResponse response
     * HttpSession session
     * Model modelModel
     * Map modelMap
     */
    @RequestMapping("/get")
    public ModelAndView get(HttpServletRequest request,HttpServletResponse response,
            HttpSession session,Model model,ModelMap modelMap){
        
        String userName=request.getParameter("userName");
        request.setAttribute("request", "request:"+userName);
        session.setAttribute("session", "session:"+userName);
        model.addAttribute("model", "model:"+userName);
        ModelAndView modelAndView = new ModelAndView("/index2.jsp");
        modelAndView.addObject("modelAndView", "modelAndView:"+userName);
        
        return modelAndView;
    }
}

  4.springmvc.xm配置文件

    ①通过BeanNameUrlHandlerMapping来映射请求和处理器之间对应的关系,通过name来进行匹配    

  <bean id="helloweb" class="test.controller.HelloWebController"></bean>
    
    <!-- 使用BeanNameUrlHandlerMapping来对处理器进行映射,这种映射关系,一个请求就对应着一个Controller类 -->
    <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean>
    <!-- 声明bean的name,因为使用了BeanNameUrlHandlerMapping,其实bean的id和name也可以当作name,也可以使用id,但是推荐使用name-->
    <bean name="/helloworld" class="test.controller.HelloWorldController"></bean>

    ②通过SimpleUrlHandlerMapping来映射   

   <!-- 使用BeanNameUrlHandlerMapping来对处理器进行映射,这种映射关系,一个请求就对应着一个Controller类 -->
    <bean class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">
        <property name="mappings">
        <!-- 配置URL与ACTION对象ID进行映射 ,<prop key="second.action">second</prop>,其中key匹配url信息,value为controller的ID -->
            <props>
                <prop key="helloweb">helloweb</prop>
                <prop key="hellojava">hellojava</prop>
            </props>
        </property>
    </bean>

    ③和上面方式结合,在一个处理器内部映射多个方法对请求进行处理

<!-- 使用BeanNameUrlHandlerMapping来对处理器进行映射,这种映射关系,一个请求就对应着一个Controller类 -->
    <bean class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">
        <property name="mappings">
        <!-- 配置URL与ACTION对象ID进行映射 ,<prop key="second.action">second</prop>,其中key匹配url信息,value为action的ID -->
            <props>
                <prop key="helloweb">helloweb</prop>
                <prop key="hellojava">hellojava</prop>
                <prop key="hellospringmvc">hellospringmvc</prop>
            </props>
        </property>
    </bean>
    
    <bean id="hellospringmvc" class="test.controller.HelloSpringMvcController" >
        <property name="methodNameResolver" ref="parameterMethodNameResolver"></property>
    </bean>
    <!-- 定义通过方法名调用控制器相关方法的规则 -->
    <bean id="parameterMethodNameResolver"
            class="org.springframework.web.servlet.mvc.multiaction.ParameterMethodNameResolver">
        <!-- 在url中使用do=方法名方式识别相关方法,例如:studentMulti.action?do=add,将调用add方法;这里的do不是固定的,可以改为其它 -->
        <property name="paramName" value="do" />
        <!-- 如果没有指定方法名时,默认 调用控制器的list方法 -->
        <property name="defaultMethodName" value="list" />
    </bean>

    ④对返回的视图加上前缀和后缀

  <!-- 支持servlet与jsp视图解析,可进行进一步处理,此步可省略 -->
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <!-- 可以加前缀或后缀 -->
        <!-- <property name="prefix" value="/jsp/"/>
        <property name="suffix" value=".jsp"/>  --> 
    </bean>

    ⑤使用注解扫描,这样就能用注解替代上面繁琐的配置,简化开发

  <!-- 开启注解扫描 -->
    <mvc:annotation-driven/>
    <context:component-scan base-package="test">
       <!-- 可以设置扫描哪些包或者注解 -->
       <!-- <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/> -->
    </context:component-scan>

  5.springMVC的拦截器   

public class MyInterceptor1 implements HandlerInterceptor {

    /**
     * 访问请求资源前执行,返回true则继续执行,返回false则不执行。
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
            throws Exception {
        System.out.println("MyInterceptor1:preHandle");
        return true;
    }

    /**
     * 访问请求资源后(但在渲染视图之前),如果没有异常,将执行此方法
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
            ModelAndView modelAndView) throws Exception {
        System.out.println("MyInterceptor1:postHandle");
    }

    /**
     * 整个请求处理完毕回调方法,即在视图渲染完毕时回调,不管理有没有异常都一定执行此方法
     */
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
            throws Exception {
        System.out.println("MyInterceptor1:afterCompletion");
    }

}
public class MyInterceptor2 implements HandlerInterceptor {

    /**
     * 访问请求资源前执行,返回true则继续执行,返回false则不执行。
     */
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
            throws Exception {
        System.out.println("MyInterceptor2:preHandle");
        return true;
    }

    /**
     * 访问请求资源后(但在渲染视图之前),如果没有异常,将执行此方法
     */
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
            ModelAndView modelAndView) throws Exception {
        System.out.println("MyInterceptor2:postHandle");
    }

    /**
     * 整个请求处理完毕回调方法,即在视图渲染完毕时回调,不管理有没有异常都一定执行此方法
     */
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
            throws Exception {
        System.out.println("MyInterceptor2:afterCompletion");
    }

}
<!-- 拦截器 -->
    <mvc:interceptors>
        <mvc:interceptor>
            <mvc:mapping path="/annotion/get"/>
            <bean class="test.Interceptor.MyInterceptor1"></bean>
        </mvc:interceptor>
        <mvc:interceptor>
            <mvc:mapping path="/annotion/get"/>
            <bean class="test.Interceptor.MyInterceptor2"></bean>
        </mvc:interceptor>
    </mvc:interceptors>

  6.springMVC异常处理

    ①实现HandlerExceptionResolver接口对异常进行处理,并将该实现类交给容器进行管理

public enum StatusCode {
    
    id_not_null(100001,"id不能为空"),
    
    success(0, "操作成功") ,
    error(9999, "服务器异常");
    
    private int status;
    private String info;
    
    private StatusCode(int status, String info) {
        this.status = status;
        this.info = info;
    }

    public int getStatus() {
        return status;
    }
    public void setStatus(int status) {
        this.status = status;
    }

    public String getInfo() {
        return info;
    }
    public void setInfo(String info) {
        this.info = info;
    }
    
}
public class ServiceException extends Exception {

    private static final long serialVersionUID = 1L;
    
    private int code;
    private String name;
    
    public int getCode() {
        return code;
    }

    public void setCode(int code) {
        this.code = code;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public ServiceException(){}
    
    public ServiceException(int code,String name){
        this.code = code;
        this.name = name;
    }

}
@Component
public class ServiceExceptionResolver implements HandlerExceptionResolver {

    @Override
    public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler,
            Exception ex) {
        ServiceException serviceException = null;
        if(ex instanceof ServiceException){
            serviceException = (ServiceException)ex;
        }else{
            serviceException = new ServiceException(StatusCode.error.getStatus(),StatusCode.error.getInfo());
            ex.printStackTrace();
        }
        
        ModelAndView mv = new ModelAndView(); 
        response.setStatus(HttpStatus.OK.value()); //设置状态码  
        response.setContentType(MediaType.APPLICATION_JSON_VALUE); //设置ContentType  
        response.setCharacterEncoding("UTF-8"); //避免乱码  
        response.setHeader("Cache-Control", "no-cache, must-revalidate");  
        try {
            response.getWriter().write("{"success":false,"code":""+serviceException.getCode()+"","msg":"" + serviceException.getName() + ""}");
        } catch (IOException e) {
            e.printStackTrace();
        }  

        return mv;  
    }

}

    ②使用@ControllerAdvice和@ExceptionHandler来对异常进行统一处理

@ControllerAdvice
public class BaseController {

    /** 基于@ExceptionHandler异常处理 */
    @ResponseBody
    @ExceptionHandler
    public Map<String, Object>  handleAndReturnData(HttpServletRequest request, HttpServletResponse response, Exception ex) {

        Map<String, Object> data = new HashMap<String, Object>();
        ServiceException e = null;
        if(ex instanceof ServiceException) {
            e = (ServiceException)ex;
        }else{
            e = new ServiceException(StatusCode.error.getStatus(),StatusCode.error.getInfo());
        }
        data.put("code", e.getCode());
        data.put("msg", e.getName());
        data.put("success", false);
        return data;
    }
}

    ③定义一个父类,并用@ExceptionHandler来标注一个方法来对异常进行处理

public class BaseController {

    /** 基于@ExceptionHandler异常处理 */
    @ResponseBody
    @ExceptionHandler
    public Map<String, Object>  handleAndReturnData(HttpServletRequest request, HttpServletResponse response, Exception ex) {

        Map<String, Object> data = new HashMap<String, Object>();
        ServiceException e = null;
        if(ex instanceof ServiceException) {
            e = (ServiceException)ex;
        }else{
            e = new ServiceException(StatusCode.error.getStatus(),StatusCode.error.getInfo());
        }
        data.put("code", e.getCode());
        data.put("msg", e.getName());
        data.put("success", false);
        return data;
    }
}
@RequestMapping("/exception")
public class ExceptionController extends BaseController{

    @RequestMapping("/info")
    public User info(User user) throws ServiceException{
        throw new ServiceException(StatusCode.id_not_null.getStatus(),StatusCode.id_not_null.getInfo());
    }
}

  

  7.文件上传下载

  

@Controller
@RequestMapping("file")
public class FileUDController {

    @ResponseBody
    @RequestMapping("/upload")
    public String upload(HttpServletRequest request,
            @RequestParam("files") MultipartFile[] files) throws Exception {

        String path = "C:/file";
        for(MultipartFile file : files){
            //如果文件不为空,写入上传路径
            if(!file.isEmpty()) {
                //上传文件路径
                //上传文件名
                String filename = file.getOriginalFilename();
                File filepath = new File(path,filename);
                //判断路径是否存在,如果不存在就创建一个
                if (!filepath.getParentFile().exists()) { 
                    filepath.getParentFile().mkdirs();
                }
                //将上传文件保存到一个目标文件当中
                file.transferTo(new File(path + File.separator + filename));
            }
        }
        return "success";
        
    }
    
    @ResponseBody
    @RequestMapping("/download")
    public String download(HttpServletRequest request,HttpServletResponse response,
            @RequestParam("fileName") String fileName) throws Exception {

        //下载文件路径,使用nio方式
        Path path = Paths.get("C:/file" + File.separator + fileName);
        response.setHeader("Content-Disposition", "attachment; filename="" + URLEncoder.encode(fileName, "UTF-8") + """);  
        response.setContentType("application/octet-stream;charset=UTF-8");  
        OutputStream outputStream = response.getOutputStream();  
        Files.copy(path, response.getOutputStream());
        outputStream.flush();  
        outputStream.close(); 
        return "success";
    }
    
    
    @RequestMapping("download2")    
    public ResponseEntity<byte[]> download(@RequestParam("fileName") String fileName) throws IOException {
        String path = "C:/file";
        File file = new File(path + File.separator + fileName);  
        HttpHeaders headers = new HttpHeaders();
        //使用URLEncoder.encode是为了解决get请求文件名乱码的问题
        headers.setContentDispositionFormData("attachment", URLEncoder.encode(fileName, "UTF-8"));   
        headers.setContentType(MediaType.APPLICATION_OCTET_STREAM);   
        return new ResponseEntity<byte[]>(FileUtils.readFileToByteArray(file),    
                                          headers, HttpStatus.CREATED);    
    }    
}
<!-- 文件上传 -->
    <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <!-- 上传文件大小上限,单位为字节(10MB) -->
        <property name="maxUploadSize">  
            <value>10485760</value>  
        </property>  
        <!-- 请求的编码格式,必须和jSP的pageEncoding属性一致,以便正确读取表单的内容,默认为ISO-8859-1 -->
        <property name="defaultEncoding">
            <value>UTF-8</value>
        </property>
    </bean> 
  
原文地址:https://www.cnblogs.com/kyleinjava/p/9106302.html