springmvc知識點

一、SpringMVC概述

Spring MVC 是由Spring官方提供的基于MVC設計理念的web框架。

SpringMVC是基于Servlet封裝的用于實現MVC控制的框架,實現前端和服務端的交互。

1.1 SpringMVC優勢

  • 嚴格遵守了MVC分層思想

  • 采用了松耦合、插件式結構;相比較于我們封裝的BaseServlet以及其他的一些MVC框架來說更靈活、更具擴展性

  • SpringMVC是基于Spring的擴展、提供了一套完善的MVC注解

  • SpringMVC在數據綁定、視圖解析都提供了多種處理方式,可靈活配置

  • SpringMVC對RESTful URL設計方法提供了良好的支持

1.2 SpringMVC本質工作

  • 接收并解析請求
  • 處理請求
  • 數據渲染、響應請求

二、SpringMVC框架部署

2.1 基于Maven創建一個web工程

  • SpringMVC是一個web框架,應用在web工程中

2.2 添加SpringMVC依賴

  • spring-context
  • spring-aspects
  • spring-jdbc
  • spring-test
  • spring-web
  • spring-webmvc
<properties><spring.version>5.2.13.RELEASE</spring.version>
</properties><dependencies><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>${spring.version}</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-aspects</artifactId><version>${spring.version}</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-jdbc</artifactId><version>${spring.version}</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-test</artifactId><version>${spring.version}</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-web</artifactId><version>${spring.version}</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId><version>${spring.version}</version></dependency></dependencies>

2.3 創建SpringMVC配置文件

  • 在resources目錄下創建名為spring-servlet.xml的文件
  • 添加MVC命名空間
<?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:context="http://www.springframework.org/schema/context"xmlns:aop="http://www.springframework.org/schema/aop"xmlns:mvc="http://www.springframework.org/schema/mvc"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--IoC采用注解配置--><context:annotation-config/><context:component-scan base-package="com.qfedu"/><!--  聲明MVC使用注解驅動  --><mvc:annotation-driven/></beans>

2.4 在web.xml中配置SpringMVC的前端控制器

SpringMVC提供了一個名為DispatcherServlet的類(SpringMVC前端控制器),用于攔截用戶請求交由SpringMVC處理

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaeehttp://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"version="3.1"><servlet><servlet-name>SpringMVC</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><init-param><param-name>contextConfigLocation</param-name><param-value>classpath:spring-servlet.xml</param-value></init-param><load-on-startup>1</load-on-startup></servlet><servlet-mapping><servlet-name>SpringMVC</servlet-name><url-pattern>/*</url-pattern></servlet-mapping></web-app>

三、SpringMVC框架使用

在SpringMVC中,我們把接收用戶請求、處理用戶請求的類稱之為Controlelr(控制器)

3.1 創建控制器

3.1.1 創建控制器類
  • 創建一個名為com.qfedu.controllers的包(包需要在Spring注解掃描的范圍內)

  • 創建一個類(無需做任何的繼承和實現)

  • 在類上添加@Controller注解聲明此類為SpringMVC的控制器

  • 在類上添加@RequestMapping("url")聲明此控制器類的請求url(可以省略)

@Controller
@RequestMapping("/book")
public class BookController {}
3.1.2 在控制器類中定義處理請求的方法
  • 在一個控制器類中可以定于多個方法處理不同的請求
  • 在每個方法上添加@RequestMapping("url")用于聲明當前方法請求的url
@Controller
@RequestMapping("/book")
public class BookController {@RequestMapping("/add")public void addBook(){System.out.println("---book akdd");}@RequestMapping("/list")public void listBooks(){System.out.println("---book list");}}
3.1.3 訪問
  • http://localhost:8080/springmvc_demo2/book/add

  • http://localhost:8080/springmvc_demo2/book/list

3.2 靜態資源配置

靜態資源:就是項目中的HTML、css、js、圖片、字體等

3.2.1 /* 和 / 的區別
  • /* 攔截所有的HTTP請求,包括.jsp的請求,都做為控制器類的請求路徑來處理
  • / 攔截所有的HTTP請求,但不包括.jsp的請求,不會放行靜態資源的請求(html/css/js/圖片)
3.2.2 靜態資源放行配置
  • 在springMVC的配置文件,添加如下靜態資源放行的配置
<!--配置靜態資源放行-->
<mvc:resources mapping="/css/**" location="/css/"/>
<mvc:resources mapping="/js/**" location="/js/"/>
<mvc:resources mapping="/imgs/**" location="/imgs/"/>
<mvc:resources mapping="/pages/**" location="/pages/"/>

3.3 前端提交數據到控制器

3.3.1 表單提交
  • 表單提交:輸入框需要提供name屬性,SpringMVC控制器是通過name屬性取值的

    <body><h3>添加圖書</h3><form action="book/add" method="post"><p>圖書名稱:<input type="text"/></p><p>圖書作者:<input type="text"/></p><p>圖書價格:<input type="text"/></p><p><input type="submit" value="提交"/></p></form>
    </body>
    
3.3.2 URL提交
  • URL提交:

    <a href="book/add?bookName=Java">URL提交</a>
    
3.3.3 AJAX提交
  • AJAX提交:請求行、請求頭、請求體都可以用來傳值

    <input type="button" value="ajax提交" id="btn1"/>
    <script type="text/javascript" src="js/jquery-3.4.1.min.js"></script>
    <script type="text/javascript">$("#btn1").click(function(){var obj = {};obj.bookName = "Java";obj.bookAuthor="張三";obj.bookPrice = 3.33;$.ajax({url:"book/add",type:"post",headers:{},contentType:"application/json",data:obj,success:function(res){console.log(res);}});});
    </script>
    

3.4 控制器接收前端提交的數據

3.4.1 @RequestParam 接收請求行傳值
  • 表單提交
  • URL提交
  • $.ajax()請求的url傳值
  • .post()/.post()/.post()/.get()中的{}傳值

**@RequestParam**注解用于接收請求行傳遞的數據

  • 前端提交數據

    <form action="book/add" method="post"><p>圖書名稱:<input type="text" name="name"/></p><p>圖書作者:<input type="text" name="author"/></p><p>圖書價格:<input type="text" name="price"/></p><p><input type="submit" value="提交"/></p>
    </form>
    
  • 控制器接收數據

    /*接收請求行數據*/
    @RequestMapping("/add")
    public void addBook(@RequestParam("name") String a,@RequestParam("author") String b,@RequestParam("price") double c){System.out.println("---book add");System.out.println(a);System.out.println(b);System.out.println(c);
    }
    

注意如果控制器方法中接收數據的參數名與請求行傳值的key一致,則@RequestParam注解可省略

@RequestMapping("/add")
public void addBook(String name,String author, double price){System.out.println("---book add");System.out.println(name);System.out.println(author);System.out.println(price);
}
3.4.2 @RequestHeader接收請求頭傳值
  • ajax封裝請求頭數據
$.ajax({...,headers:{},...
})

**@RequestHeader**注解用于接收請求行頭傳遞的數據

  • 前端

    <input type="button" value="ajax提交" id="btn1"/><script type="text/javascript" src="js/jquery-3.4.1.min.js"></script><script type="text/javascript">$("#btn1").click(function(){$.ajax({url:"book/list",type:"post",headers:{token:"wahahaawahaha"},success:function(res){console.log(res);}});});</script>
    
  • 控制器

    @RequestMapping("/list")
    public void listBooks(@RequestHeader("token") String token){System.out.println("---book list");
    }
    
3.4.3 @RequestBody接收請求體傳值
  • ajax封裝請求體數據

    $.ajax({...,contentType:"application/json",data:obj,,...
    })
    

**@RequestBody**注解用于接收請求行頭傳遞的數據

  • 前端

    <input type="button" value="ajax提交" id="btn1"/><script type="text/javascript" src="js/jquery-3.4.1.min.js"></script><script type="text/javascript">$("#btn1").click(function(){var obj = {};obj.bookName = "Python";obj.bookAuthor="杰哥";obj.bookPrice = 2.22;var s = JSON.stringify(obj); //將對象轉換成JSON格式$.ajax({url:"book/update",type:"post",contentType:"application/json",data:s,   //如果data的值為json格式字符串,contentType必須設置為"application/json"success:function(res){console.log(res);}});});</script>
    

@RquestBody 將前端請求體提交的JSON格式數據轉換成Java對象,依賴jackson包

  • 導入jackson的依賴

    <dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>2.12.1</version>
    </dependency>
    
  • 控制器

    @RequestMapping("/update")
    public void update(@RequestBody Book book){System.out.println("---book update");System.out.println(book);
    }
    

3.5 控制器響應前端請求

3.5.1 控制器響應同步請求

同步請求:form、超鏈接

處理同步請求的方法的返回類型定義為String或者ModelAndView,以實現頁面的跳轉

  • 返回類型為String

    轉發

    @RequestMapping("/add")
    public String addBook(String name, String author, double price){System.out.println("---book add");return "/tips.jsp";
    }
    

    重定向

    @RequestMapping("/add")
    public String addBook(String name, String author, double price){System.out.println("---book add");return "redirect:/tips.jsp";
    }
    
  • 返回類型為 ModelAndView

    轉發

    @RequestMapping("/add")
    public ModelAndView addBook(String name, String author, double price){System.out.println("---book add");ModelAndView modelAndView = new ModelAndView("/tips.jsp");return modelAndView;
    }
    

    重定向

    @RequestMapping("/add")
    public ModelAndView addBook(String name, String author, double price){System.out.println("---book add");ModelAndView modelAndView = new ModelAndView("redirect:/tips.jsp");return modelAndView;
    }
    
3.5.2 控制器響應異步請求

異步請求:ajax請求

使用response中的輸出流進行響應

  • 控制器方法的返回類型為void
  • 控制器方法添加 HttpServletResponse response參數
  • 在方法中通過response獲取輸出流,使用流響應ajax請求
@RequestMapping("/update")
public void update(@RequestBody Book book, HttpServletResponse response) throws IOException {System.out.println("---book update");System.out.println(book);//使用ObjectMapper將對象轉換成JSON格式字符串String s = new ObjectMapper().writeValueAsString(book);response.setCharacterEncoding("utf-8");response.setContentType("application/json");PrintWriter out = response.getWriter();out.println(s);out.flush();out.close();
}

直接在控制器方法返回響應的對象

  • 控制器方法的返回類型設置為響應給ajax請求的對象類型
  • 在控制器方法前添加@ResponseBody注解,將返回的對象轉換成JSON響應給ajax請求
  • 如果一個控制器類中的所有方法都是響應ajax請求,則可以直接在控制器類前添加@ResponseBody注解
@RequestMapping("/update")
@ResponseBody
public List<Book> update() {System.out.println("---book update");List<Book> books = new ArrayList<Book>();books.add(new Book(1,"Java","老張",2.22));books.add(new Book(2,"C++","老李",3.22));return books;
}
3.5.3 控制器響應同步請求的數據傳遞

對于同步請求的轉發響應,我們可以傳遞參數到轉發的頁面

  • 返回類型為String:

    //1.在控制器方法中定義一個Model類型的參數
    //2.在return頁面之前,向model中添加鍵值對,添加的鍵值對就會被傳遞到轉發的頁面
    @RequestMapping("/add")
    public String addBook(String name, String author, double price,Model model){model.addAttribute("key1","value1");model.addAttribute("book",new Book(1,"Java","老張",2.22));return "/tips.jsp";
    }//除了使用Model對象傳值外,還可以直接使用HttpServletRequest對象
    @RequestMapping("/add")
    public String addBook(String name, String author, double price,HttpServletRequest request){request.setAttribute("key1","value1");request.setAttribute("book",new Book(1,"Java","老張",2.22));return "/tips.jsp";
    }
    
  • 返回類型ModelAndView:

    @RequestMapping("/add2")
    public ModelAndView addBook2(String name, String author, double price){ModelAndView modelAndView = new ModelAndView("/tips.jsp");modelAndView.addObject("key1","value1");modelAndView.addObject("book",new Book(1,"Java","老張",2.22));return modelAndView;
    }
    

3.6 解決中文亂碼問題

3.6.1 前端編碼
  • JSP頁面:

    <%@ page contentType="text/html;charset=UTF-8" language="java" pageEncoding="UTF-8" %>
    
  • HTML頁面:

    <meta charset="UTF-8">
    
3.6.2 服務器編碼
  • tomcat/conf/server.xml

    <Connector port="8080" protocol="HTTP/1.1"connectionTimeout="20000"redirectPort="8443"  URIEncoding="UTF-8"/>
    
3.6.3 設置SpringMVC的編碼方式
  • 在web.xml中配置SpringMVC編碼過濾器的編碼方式

    <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>
    

四、SpringMVC的請求處理流程

4.1 請求處理流程

SpringMVC通過前端控制器(DispatcherServlet)攔截并處理用戶請求的

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-PrhcJROA-1639494012800)(imgs/1617257389694.png)]

① 前端發送請求被前端控制器DispatcherServlet攔截
② 前端控制器調用處理器映射器HandlerMapping對請求URL進行解析,解析之后返回調用給前端控制器
③ 前端控制器調用處理器適配器處理調用鏈
④ 處理器適配器基于反射通過適配器設計模式完成處理器(控制器)的調用處理用戶請求
⑤ 處理器適配器將控制器返回的視圖和數據信息封裝成ModelAndView響應給前端控制器
⑥ 前端控制器調用視圖解析器ViewResolver對ModelAndView進行解析,將解析結果(視圖資源和數據)響應給前端控制器
⑦ 前端控制器調用視圖view組件將數據進行渲染,將渲染結果(靜態視圖)響應給前端控制器
⑧ 前端控制器響應用戶請求

4.2 SpringMVC的核心組件

  • DispatcherServlet 前端控制器、總控制器

    • 作用:接收請求,協同各組件工作、響應請求
  • HandlerMapping 處理器映射

    • 作用:負責根據用戶請求的URL找到對應的Handler
    • 可配置 SpringMVC提供多個處理器映射的實現,可以根據需要進行配置
  • HandlerAdapter 處理器適配器

    • 作用:按照處理器映射器解析的用戶請求的調用鏈,通過適配器模式完成Handler的調用
  • Handler 處理器/控制器

    • 由工程師根據業務的需求進行開發
    • 作用:處理請求
  • ModelAndView 視圖模型

    • 作用:用于封裝處理器返回的數據以及相應的視圖
    • ModelAndView = Model + View
  • ViewResolver 視圖解析器

    • 作用:對ModelAndView進行解析
    • 可配置 SpringMVC提供多個視圖解析器的實現,可以根據需要進行配置
  • View 視圖

    • 作用:完成數據渲染

4.3 處理器映射器

不同的處理器映射器對URL處理的方式也不相同,使用對應的處理器映射器之后我們的前端請求規則也需要發生相應的變化

SpringMVC提供的處理器映射器:

  • BeanNameUrlHandlerMapping 根據控制器的ID訪問控制器
  • SimpleUrlHandlerMapping 根據控制器配置的URL訪問(默認)

配置處理器映射器:

  • 在SpringMVC的配置文件中通過bean標簽聲明處理器映射器

  • 配置BeanNameUrlHandlerMapping

    <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean>
    
  • 配置SimpleUrlHandlerMapping

    <bean class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping"><property name="mappings"><props><prop key="/aaa">bookController</prop><prop key="/bbb">studentController</prop></props></property>
    </bean>
    

4.4 視圖解析器

Spring提供了多個視圖解析器:

  • UrlBasedViewResolver
  • InternalResourceViewResolver
  • UrlBasedViewResolver 需要依賴jstl

    • 添加JSTL的依賴
    <dependency><groupId>javax.servlet</groupId><artifactId>jstl</artifactId><version>1.2</version>
    </dependency>
    
    • 配置視圖解析器
    <bean id="viewResolver" class="org.springframework.web.servlet.view.UrlBasedViewResolver"><property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/><property name="prefix" value="/"/><property name="suffix" value=".jsp"/>
    </bean>
    
  • InternalResourceViewResolver

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

五、日期格式處理

5.1 在控制器中使用對象接收數據

  • 前端:

    <form action="test/add" method="post"><p>圖書名稱:<input type="text" name="bookName"/></p><p>圖書作者:<input type="text" name="bookAuthor"/></p><p>圖書價格:<input type="text" name="bookPrice"/></p><p><input type="submit" value="提交"/></p>
    </form>
    
  • 后端

    @Controller
    @RequestMapping("/test")
    public class TestController {@RequestMapping("/add")//表單提交的多個數據,在控制器方法中可以使用對象接收//但是提交的數據的key必須要與對象的屬性名一致public String addBook(Book book){return "/tips.jsp";}}
    

5.2 日期格式處理

如果前端需要輸入日期數據,在控制器中轉換成Date對象,SpringMVC要求前端輸入的日期格式必須為yyyy/MM/dd

如果甲方要求日期格式必須為指定的格式,而這個指定格式SpringMVC不接受,該如何處理呢?

  • 自定義日期轉換器
5.2.1 創建自定義日期轉換器
/**** 1.創建一個類實現Converter接口,泛型指定從什么類型轉換為什么類型* 2.實現convert轉換方法*/
public class MyDateConverter implements Converter<String, Date> {SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日");public Date convert(String s) {Date date = null;try {date = sdf.parse(s);} catch (ParseException e) {e.printStackTrace();}return date;}}
5.2.2 配置自定義轉換器
<!--  聲明MVC使用注解驅動  -->
<mvc:annotation-driven conversion-service="converterFactory"/><bean id="converterFactory" class="org.springframework.format.support.FormattingConversionServiceFactoryBean"><property name="converters"><set><bean class="com.qfedu.utils.MyDateConverter"/></set></property>
</bean>

六、文件上傳下載

6.1 SpringMVC框架部署

  • 基于Maven創建web工程

  • 添加SpringMVC所需的依賴

    • Spring:context aspects jdbc test web webmvc jackson
  • 創建SpringMVC配置文件

  • 在web.xml中配置SpringMVC的前端控制器

  • 在web.xml中配置SpringMVC的編碼過濾器

  • 配置springmvc靜態資源處理策略

6.2 文件上傳

案例:添加圖書,同時提交圖書的封面圖片

6.2.1 前端提交文件
  • 表單提示方式必須為post
  • 表單enctype屬性設置為multipart/form-data
<form action="book/add" method="post" enctype="multipart/form-data"><p>圖書名稱:<input type="text" name="bookName"/></p><p>圖書作者:<input type="text" name="bookAuthor"/></p><p>圖書價格:<input type="text" name="bookPrice"/></p><p>圖書封面:<input type="file" name="imgFile"/></p><p><input type="submit" value="提交"/></p>
</form>
6.2.2 控制器接收數據和文件

SpringMVC處理上傳文件需要借助于CommonsMultipartResolver文件解析器

  • 添加依賴:commons-io commons-fileupload

    <dependency><groupId>commons-io</groupId><artifactId>commons-io</artifactId><version>2.4</version>
    </dependency>
    <dependency><groupId>commons-fileupload</groupId><artifactId>commons-fileupload</artifactId><version>1.4</version>
    </dependency>
    
  • 在spring-servlet.xml中配置文件解析器

    <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"><property name="maxUploadSize" value="10240000"/><property name="maxInMemorySize" value="102400"/><property name="defaultEncoding" value="utf-8"/>
    </bean>
    
  • 控制器接收文件

    • 在處理文件上傳的方法中定義一個MultiPartFile類型的對象,就可以接受圖片了
    @Controller
    @RequestMapping("/book")
    public class BookController {@RequestMapping("/add")public String addBook(Book book, MultipartFile imgFile, HttpServletRequest request) throws IOException {System.out.println("--------------add");//imgFile就表示上傳的圖片//1.截取上傳文件的后綴名,生成新的文件名String originalFilename = imgFile.getOriginalFilename();String ext = originalFilename.substring( originalFilename.lastIndexOf(".") ); String fileName = System.currentTimeMillis()+ext;//2.獲取imgs目錄在服務器的路徑String dir = request.getServletContext().getRealPath("imgs");String savePath = dir+"/"+fileName; //3.保存文件imgFile.transferTo( new File(savePath));//4.將圖片的訪問路徑設置到book對象book.setBookImg("imgs/"+fileName);//5.調用service保存book到數據庫return "/tips.jsp";}}
    

6.3 文件下載

6.3.1 顯示文件列表
  • list.jsp

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html><head><title>Title</title></head><!-- 最新版本的 Bootstrap 核心 CSS 文件 --><link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@3.3.7/dist/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous"><!-- 可選的 Bootstrap 主題文件(一般不用引入) --><link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@3.3.7/dist/css/bootstrap-theme.min.css" integrity="sha384-rHyoN1iRsVXV4nD0JutlnGaslCJuC7uwjduW9SVrLvRYooPp2bWYgmgJQIXwl/Sp" crossorigin="anonymous"><!-- 最新的 Bootstrap 核心 JavaScript 文件 --><script src="https://cdn.jsdelivr.net/npm/bootstrap@3.3.7/dist/js/bootstrap.min.js" integrity="sha384-Tc5IQib027qvyjSMfHjOMaLkfuWVxZxUPnCJA7l2mCWNIpG9mGCD8wGNIcPD7Txa" crossorigin="anonymous"></script><body><h4>文件列表</h4><div class="row" id="container"></div><script type="text/javascript" src="js/jquery-3.4.1.min.js"></script><script type="text/javascript">$.get("book/list",function(res){for (var i = 0; i < res.length; i++) {var fn = res[i];var htmlStr = "<div class='col-lg-2  col-md-3 col-sm-4 col-xs-6'><div class='thumbnail'><img src='imgs/"+fn+"' alt='...'> <div class='caption'><p><a href='#' class='btn btn-primary' role='button'>下載</a></p></div> </div> </div>";$("#container").append(htmlStr);}},"json");</script></body>
    </html>
    
  • BookController

    @RequestMapping("/list")
    @ResponseBody
    public String[] listImgs(HttpServletRequest request){//從imgs目錄下獲取所有的圖片信息String dir = request.getServletContext().getRealPath("imgs");File imgDir = new File(dir);String[] fileNames = imgDir.list();return fileNames;
    }
    
6.3.2 實現文件下載
  • list.jsp

    var htmlStr = "<div class='col-lg-2  col-md-3 col-sm-4 col-xs-6'><div class='thumbnail'><img src='imgs/"+fn+"' alt='...'> <div class='caption'><p><a href='book/download?fname="+fn+"' class='btn btn-primary' role='button'>下載</a></p></div> </div> </div>";
    
  • BookController

    @RequestMapping("/download")
    public void downloadImg(String fname, HttpServletRequest request, HttpServletResponse response) throws Exception {//從imgs目錄找到當前文件String dir = request.getServletContext().getRealPath("imgs");String filePath = dir+"/"+fname;FileInputStream fileInputStream = new FileInputStream(filePath);response.setContentType("application/exe");response.addHeader("Content-Disposition","attachment;filename="+fname);IOUtils.copy(fileInputStream, response.getOutputStream());
    }
    

七、統一異常處理

在我們的應用系統運行的過程中,可能由于運行環境、用戶操作、資源不足等各方面的原因導致系統出現異常(HTTP狀態異常、Exception);如果系統出現了異常,這些異常將會通過瀏覽器呈現給用戶,而這種異常的顯示是沒有必要,因此我們可以在服務器進行特定的處理——當系統出現異常之后,呈現給用戶一個統一的、可讀的的異常提示頁面。

7.1 HTTP異常狀態統一處理

HTTP Status 404

  • 創建一個用于進行異常提示的頁面:404.jsp

  • 在web.xml中進行配置:

    <error-page><error-code>404</error-code><location>/404.jsp</location>
    </error-page>
    

7.2 Java代碼異常的統一處理

7.2.1 基于Servlet-api的處理
  • 創建異常提示頁面:err.jsp

  • 在web.xml中進行配置

    <error-page><exception-type>java.lang.NumberFormatException</exception-type><location>/err.jsp</location>
    </error-page>
    
7.2.2 SpringMVC處理
  • 使用異常處理類進行統一處理

    @ControllerAdvice
    public class MyExceptionHandler {@ExceptionHandler(NullPointerException.class)public String nullHandler(){return "/err1.jsp";}@ExceptionHandler(NumberFormatException.class)public String formatHandler(){return "/err2.jsp";}}
    

八、攔截器

8.1 攔截器介紹

SpringMVC提供的攔截器就類似于Servlet-api中的過濾器,可以對控制器的請求進行攔截實現相關的預處理和后處理。

  • 過濾器
    • 是Servlet規范的一部分,所有的web項目都可以使用
    • 過濾器在web.xml配置(可以使用注解),能夠攔截所有web請求
  • 攔截器
    • 是SpringMVC框架的實現,只有在SpringMVC框架中才能使用
    • 攔截器在SpringMVC配置文件進行配置,不會攔截SpringMVC放行的資源(jsp\html\css…)

8.2 自定義攔截器

8.2.1 創建攔截器
public class MyInterceptor1 implements HandlerInterceptor {//預處理方法public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println("--------------預處理");Enumeration<String> keys = request.getParameterNames();while (keys.hasMoreElements()){String key = keys.nextElement();if("bookId".equals(key)){return true;}}response.setStatus(400);return false;}//后處理方法public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {modelAndView.addObject("tips","這是通過攔截器的后處理添加的數據");System.out.println("--------------后處理");}
}
8.2.2 配置攔截器
<mvc:interceptors><mvc:interceptor><mvc:mapping path="/book/query"/><mvc:mapping path="/book/add"/><mvc:mapping path="/student/**"/><mvc:exclude-mapping path="/student/add"/><bean class="com.qfedu.utils.MyInterceptor1"/></mvc:interceptor>
</mvc:interceptors>

8.3 攔截器鏈

將多個攔截器按照一定的順序構成一個執行鏈

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-XHv4rqsX-1639494012803)(imgs/1617334128888.png)]

九、SSM整合

9.1 創建web項目

  • 創建maven工程

  • 修改pom文件

    <packaging>war</packaging>
    
  • 完成maven工程web項目結構

  • 添加web項目依賴:

    <dependency><groupId>javax.servlet</groupId><artifactId>jsp-api</artifactId><version>2.0</version><scope>provided</scope>
    </dependency><dependency><groupId>javax.servlet</groupId><artifactId>javax.servlet-api</artifactId><version>4.0.1</version><scope>provided</scope>
    </dependency>
    
  • 配置服務器運行環境

9.2 部署MyBatis

  • 添加MyBatis依賴

    <!-- mysql驅動 -->
    <dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>5.1.47</version>
    </dependency><!-- mybatis依賴 -->
    <dependency><groupId>org.mybatis</groupId><artifactId>mybatis</artifactId><version>3.4.6</version>
    </dependency><!-- lombok -->
    <dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><version>1.18.16</version><scope>provided</scope>
    </dependency>
    
  • 創建MyBatis配置文件mybatis-config.xml

    <?xml version="1.0" encoding="UTF-8" ?>
    <!DOCTYPE configuration PUBLIC "-//mybatis.org//DTD Config 3.0//EN""http://mybatis.org/dtd/mybatis-3-config.dtd">
    <configuration></configuration>
    

9.3 部署Spring、SpringMVC

9.3.1 添加依賴
<properties><spring.version>5.2.13.RELEASE</spring.version>
</properties><!--context-->
<dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>${spring.version}</version>
</dependency>
<!--aspects-->
<dependency><groupId>org.springframework</groupId><artifactId>spring-aspects</artifactId><version>${spring.version}</version>
</dependency>
<!--jdbc-->
<dependency><groupId>org.springframework</groupId><artifactId>spring-jdbc</artifactId><version>${spring.version}</version>
</dependency>
<!--test-->
<dependency><groupId>org.springframework</groupId><artifactId>spring-test</artifactId><version>${spring.version}</version>
</dependency>
<!--web-->
<dependency><groupId>org.springframework</groupId><artifactId>spring-web</artifactId><version>${spring.version}</version>
</dependency>
<!--webmvc-->
<dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId><version>${spring.version}</version>
</dependency>
<!--jackson-databind-->
<dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>2.12.1</version>
</dependency>
9.3.2 創建Spring配置
  • 多配置文件分開配置

    • spring-context.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:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsd"><!-- 聲明使用注解配置 --><context:annotation-config/><!-- 聲明Spring工廠注解的掃描范圍 --><context:component-scan base-package="com.qfedu"/></beans>
      
    • spring-mvc.xml 進行mvc相關的配置,例如靜態資源配置、攔截器配置等

      <?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:mvc="http://www.springframework.org/schema/mvc"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/mvchttp://www.springframework.org/schema/mvc/spring-mvc.xsd"><!--聲明MVC使用注解配置--><mvc:annotation-driven/></beans>
      
    • spring-mybatis.xml 進行Spring與MyBatis整合相關的配置

      <beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"xmlns:tx="http://www.springframework.org/schema/tx"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsdhttp://www.springframework.org/schema/txhttp://www.springframework.org/schema/tx/spring-tx.xsd"></beans>
      
9.3.3 配置SpringMVC前端控制器
  • 在web.xml進行配置
<servlet><servlet-name>springmvc</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><init-param><param-name>contextConfigLocation</param-name><param-value>classpath:spring-*.xml</param-value></init-param><load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping><servlet-name>springmvc</servlet-name><url-pattern>/</url-pattern>
</servlet-mapping>

9.4 整合配置(IoC)

9.4.1 導入mybatis-spring依賴
<!-- https://mvnrepository.com/artifact/org.mybatis/mybatis-spring -->
<dependency><groupId>org.mybatis</groupId><artifactId>mybatis-spring</artifactId><version>1.3.2</version>
</dependency>
9.4.2 配置druid連接池
  • 添加druid依賴

    <dependency><groupId>com.alibaba</groupId><artifactId>druid</artifactId><version>1.1.10</version>
    </dependency>
    
  • 創建druid.properties,并配置:

    druid.driver=com.mysql.jdbc.Driver
    druid.url=jdbc:mysql://localhost:3306/db_2010_mybatis?characterEncoding=utf-8
    druid.username=root
    druid.password=admin123## 連接池參數
    druid.pool.init=1
    druid.pool.minIdle=3
    druid.pool.maxActive=20
    druid.pool.timeout=30000
    
  • 在spring-mybatis.xml配置數據源

    <context:property-placeholder location="classpath:druid.properties"/><bean id="druidDataSource" class="com.alibaba.druid.pool.DruidDataSource"><property name="driverClassName" value="${druid.driver}"/><property name="url" value="${druid.url}"/><property name="username" value="${druid.username}"/><property name="password" value="${druid.password}"/><property name="initialSize" value="${druid.pool.init}"/><property name="minIdle" value="${druid.pool.minIdle}"/><property name="maxActive" value="${druid.pool.maxActive}"/><property name="maxWait" value="${druid.pool.timeout}"/>
    </bean>
    
9.4.3 配置SqlSessionFactory
  • 在spring-mybatis.xml配置

    <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"><property name="dataSource" ref="druidDataSource"/><property name="mapperLocations" value="classpath:mappers/*.xml"/><property name="typeAliasesPackage" value="com.qfedu.bean"/><property name="configLocation" value="classpath:mybatis-config.xml"/>
    </bean>
    
9.4.4 配置MapperScannerConfigurer
  • 在spring-mybatis.xml配置

    <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"><property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/><property name="basePackage" value="com.qfedu.dao"/>
    </bean>
    

9.5 整合配置(AOP)

使用Spring提供的事務管理完成DAO操作的事務管理

基于注解的事務管理配置:

  • 將Spring提供的事務管理切面類配置到Spring容器

    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"><property name="dataSource" ref="druidDataSource"/>
    </bean><tx:annotation-driven transaction-manager="transactionManager"/>
    

9.6 整合測試

9.6.1 完成User的查詢操作
  • 創建實體類

    @Data
    @NoArgsConstructor
    @AllArgsConstructor
    @ToString
    public class User {private int userId;private String userName;private String userPwd;private String userRealname;private String userImg;}
    
  • 在DAO包中創建接口

    public interface UserDAO {public User queryUserByName(String name);}
    
  • 在mappers目錄下創建映射文件

    <?xml version="1.0" encoding="UTF-8" ?>
    <!DOCTYPE mapperPUBLIC "-//mybatis.org//DTD Mapper 3.0//EN""http://mybatis.org/dtd/mybatis-3-mapper.dtd">
    <mapper namespace="com.qfedu.dao.UserDAO"><resultMap id="userMap" type="User"><id column="user_id" property="userId"/><result column="user_name" property="userName"/><result column="user_pwd" property="userPwd"/><result column="user_realname" property="userRealname"/><result column="user_img" property="userImg"/></resultMap><select id="queryUserByName" resultMap="userMap">select user_id,user_name,user_pwd,user_realname,user_imgfrom userswhere user_name=#{userName}</select></mapper>
    
9.6.2 對DAO單元測試
  • 添加junit、spring-test依賴

     <dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version><scope>test</scope>
    </dependency>
    
  • 創建測試類

    @RunWith(SpringJUnit4ClassRunner.class)
    @ContextConfiguration({"classpath:spring-context.xml","classpath:spring-mvc.xml","classpath:spring-mybatis.xml"})
    public class UserDAOTest {@Resourceprivate UserDAO userDAO;@Testpublic void queryUserByName() {User user = userDAO.queryUserByName("wangwu");System.out.println(user);}
    }
    

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/450423.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/450423.shtml
英文地址,請注明出處:http://en.pswp.cn/news/450423.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

spring @component的作用

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 1、controller 控制器&#xff08;注入服務&#xff09; 2、service 服務&#xff08;注入dao&#xff09; 3、repository dao&#xff…

微信小程序 懸浮按鈕

2019獨角獸企業重金招聘Python工程師標準>>> 效果視頻 https://pan.baidu.com/s/1yfrDaG9YAX0--v0EA3awZA 布局需要按照圓形排列&#xff0c;所以我們需要計算每個點的坐標 代碼部分 <view styleposition:fixed; wx:for"{{list}}" wx:for-index"i…

C語言const關鍵字—也許該被替換為readolny

const 是constant 的縮寫&#xff0c;是恒定不變的意思&#xff0c;也翻譯為常量、常數等。很不幸&#xff0c;正是因為這一點&#xff0c;很多人都認為被const 修飾的值是常量。這是不精確的&#xff0c;精確的說應該是只讀的變量&#xff0c;其值在編譯時不能被使用&#xff…

dbus服務自啟動方法

Linux 一般發行版上 "/usr/share/dbus-1/services/"目錄就是dbus放service文件的地方。 需要自動啟動的服務器 就在這個目錄放一個 service文件&#xff0c;內容如下&#xff1a; $ cat /usr/share/dbus-1/services/dhcdbd.service [D-BUS Service] Namecom.redhat.…

在Spring Boot中使用 @ConfigurationProperties 注解

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 上一篇博客寫了 如何在Spring Boot application中配置mail . 使用 Value 注解注入屬性. 但 Spring Boot 提供了另一種方式 &#xff0c;能…

Micronaut教程:如何使用基于JVM的框架構建微服務

\本文要點\\Micronaut是一種基于jvm的現代化全棧框架&#xff0c;用于構建模塊化且易于測試的微服務應用程序。\\tMicronaut提供完全的編譯時、反射無關的依賴注入和AOP。\\t該框架的開發團隊和Grails框架的開發團隊是同一個。\\tMicronaut框架集成了云技術&#xff0c;服務發現…

C語言extern關鍵詞—最會帶帽子的關鍵字

extern&#xff0c;外面的、外來的意思。那它有什么作用呢&#xff1f;舉個例子&#xff1a;假設你在大街上看到一個黑皮膚綠眼睛紅頭發的美女&#xff08;外星人&#xff1f;&#xff09;或者帥哥。你的第一反應就是這人不是國產的。extern 就相當于他們的這些區別于中國人的特…

解決Coldfusion連接MySQL數據庫的問題

在連接MySQL時&#xff0c;出現了如下錯誤&#xff1a; Connections to MySQL Community Server are not supported. Please contact MySQL to obtain a MySQL Enterprise or Commercial version. 解決方案&#xff1a; step 1: download the JDBC driver JDBC Driver for MySQ…

范式知識點

Mysql數據庫 ?前關系數據庫有六種范式&#xff1a; 第?范式&#xff08;1NF&#xff09;、第?范式&#xff08;2NF&#xff09;、第三范式&#xff08;3NF&#xff09;、巴斯-科德范式 &#xff08;BCNF&#xff09;、第四范式(4NF&#xff09;和第五范式&#xff08;5NF&a…

Tensorflow入門----占位符、常量和Session

安裝好TensorFlow之后&#xff0c;開一個python環境&#xff0c;就可以開始運行和使用TensorFlow了。 先給一個實例&#xff0c; #先導入TensorFlowimport tensorflow as tf # Create TensorFlow object called hello_constanthello_constant tf.constant(Hello World!) with …

C語言union關鍵字

union 關鍵字的用法與struct 的用法非常類似。union 維護足夠的空間來置放多個數據成員中的“一種”&#xff0c;而不是為每一個數據成員配置空間&#xff0c;在union 中所有的數據成員共用一個空間&#xff0c;同一時間只能儲存其中一個數據成員&#xff0c;所有的數據成員具有…

js 深拷貝 和 淺拷貝

1、 ...運算符 &#xff08;淺拷貝&#xff09; let obj {a:1,b:2}; let obj2 {...obj}; obj.a3 obj //{a: 3, b: 2} obj2 //{a: 1, b: 2}a {a:[{b:1}]}; b a ;b.a[0].b 2; //b {a:[{b:2}]}a // a {a:[{b:2}]} 2、 JSON.parse(JSON.stringify(參數))…

軟考如何備考

軟考上半年 報名時間 3月 考試時間 5月 備考持續時間&#xff1a;一個半月&#xff0c;每天至少兩個小時。 第一輪 基礎知識輪&#xff0c; 15天 在這一輪中如果碰到無法理解的知識點一定不要過于糾結&#xff0c;往后學&#xff0c;一般后面的學著學著之前的內容就…

Python學習第一天-第2節

*本節課內容參見&#xff1a;https://www.cnblogs.com/jin-xin/articles/7459977.html運行第一個Python程序 print(hello world) 將文本文件保存為以.py結尾的文件&#xff0c;如hello_world.py &#xff0c;在終端中&#xff0c;運行命令&#xff1a;Python hello_world.py&am…

SpringMVC之組合注解@GetMapping

前些天發現了一個巨牛的人工智能學習網站&#xff0c;通俗易懂&#xff0c;風趣幽默&#xff0c;忍不住分享一下給大家。點擊跳轉到教程。 Spring4.3中引進了&#xff5b;GetMapping、PostMapping、PutMapping、DeleteMapping、PatchMapping&#xff5d;&#xff0c;來幫助簡化…

資源下載地址

1、nginx http://nginx.org/en/download.html 2、nodejs http://nodejs.cn/download/ 3、vscode https://code.visualstudio.com/ 4、canal https://github.com/alibaba/canal/releases/download/canal-1.1.4/canal.deployer-1.1.4.tar.gz 5、jenkins https://jenkins.io/i…

C語言volatile關鍵字—最易變的關鍵字

volatile 是易變的、不穩定的意思。很多人根本就沒見過這個關鍵字&#xff0c;不知道它的存在。也有很多程序員知道它的存在&#xff0c;但從來沒用過它。我對它有種“楊家有女初長成,養在深閨人未識” 的感覺。volatile 關鍵字和const 一樣是一種類型修飾符&#xff0c;用它修…

mysql 備份腳本

#!/bin/bash INNOBACKUPEXFULL"/usr/bin/innobackupex" MYSQL_CMD --socket/ssd/mysql/3346/tmp/mysql.sock --userroot --passwordmysqlpassword --port3346 MYSQL_UP --userroot --passwordmysqlpassword --port3346 #mysqladmin的用戶名和密碼 TMPLOG"/dat…

爆棧的處理方法

爆棧指遞歸中&#xff0c;存儲的信息量大于系統棧的內存。 信息量包括元素編號&#xff0c;每一層中開的變量。 和遞歸的層數正相關。 &#xff08;雖然noip一般開棧&#xff09; 1.手寫棧 while(top){ int xsta[top]; for(each son) if(has son){ //blablabla sta[top]son; h…

項目面試題

項目面試題 1.描述最熟悉的或者&#xff08;最近&#xff09;的一個項目 &#xff1f; 目的&#xff1a;檢測技術和項目的熟悉程度工作職責 回答&#xff1a;項目名稱&#xff0c;項目的甲方&#xff0c;背景&#xff0c;實現的目的和意義&#xff0c;該項目主要是為了解決什…