文章目錄
- 環境
- 1 Mybatis
- 1.1 資料庫組態檔
- 1.2 配置 Mybatis
- 2 Spring 整合 Mybatis
- 2.1 Spring 整合 dao 層
- 2.1.1 context:property-placeholder標簽作用?
- 2.1.2 資料庫連接池
- 2.1.3 配置 SqlSessionFactory 物件
- 2.1.4 配置掃描dao介面包
- 2.2 Spring 整合 service 層
- 3 整合 SpringMVC
- 3.1 web.xml 檔案是如何執行的?
- 3.1.1 執行順序
- 3.1.2 init-param標簽
- 3.1.3 load-on-startup標簽
- 3.2 springmvc 組態檔是如何執行的?
- 3.2.1 mvc:annotation-driven作用?
- 3.2.2 mvc:default-servlet-handler作用?
- 3.2.3 context:component-scan 作用?
- 3.2.4 視圖決議器 作用?
- 4 Spring 配置整合檔案
- 結束
環境
- MySQL 8.0.16
1 Mybatis
1.1 資料庫組態檔
jdbc.driver=com.mysql.cj.jdbc.Driver
# 如果使用mysql 6+,增加一個時區的配置
jdbc.url=jdbc:mysql://localhost:3306/ssmbuild?useUnicode=true&characterEncoding=utf8&serverTimezone=UTC&useSSL=false
jdbc.username=root
jdbc.password=123456
com.mysql.jdbc.Driver 與 com.mysql.cj.jdbc.Driver ?
- com.mysql.jdbc.Driver 是 mysql-connector-java 5 中的
- com.mysql.cj.jdbc.Driver 是 mysql-connector-java 6 中的
JDBC 連接 mysql 5:
url=jdbc:mysql://localhost:3306/test?useUnicode=true&characterEncoding=utf8&useSSL=false
JDBC 連接 mysql 6:
url=jdbc:mysql://localhost:3306/test?serverTimezone=UTC&?useUnicode=true&characterEncoding=utf8&useSSL=false
1.2 配置 Mybatis
<?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>
<!--配置資料源,交給 spring 去做-->
<typeAliases>
<package name="com.fyy.pojo"/>
</typeAliases>
<mappers>
<mapper class="com.fyy.dao.BookMapper"/>
</mappers>
</configuration>
typeAiases 標簽作用?
其中,<typeAiases>標簽的作用只是為 Java 型別指定一個短的名字,它只和xml 配置有關,存在的意義僅在于用來減少類完全限定名的冗余,
指定一個包名,Mybatis 會在此包名下搜索需要的 Java Bean,每一個在此包下的 Java Bean,在沒有注解的情況下,會使用 Bean 的首字母小寫的非限定類名作為它的別名,比如:com.fyy.pojo.User 的別名為:user ,若有注解,則別名為其注解值,如下例子:
@Alias("hello")
public class Hello(){}
mappers標簽作用?
Mybatis 是基于 Sql 映射配置的框架,Sql 陳述句在Mapper 組態檔中,當構建 SqlSession 類之后,就需要去讀取 Mapper 組態檔中的 sql 配置,
mappers : 映射器,以最佳的方式告訴 Mybatis 去哪里找映射檔案,就是用來配置需要加載的 sql 映射組態檔路徑的,
mappers 下的每一個 mapper 都是一個mapper,配置的都是一個獨立的映射組態檔的路徑,配置方式有以下幾種,
1、介面所在包
<mappers>
<!-- mapper介面所在的包名 -->
<package name="com.fyy.mapper"/>
</mappers>
# package標簽,通過 name 屬性指定 mapper 介面所在的包名,此時對應的映射檔案必須
與介面位于同一路徑下,并且名稱相同,
2、相對路徑配置
<mappers>
<mapper resource="com/fyy/mapper/FlowerMapper.xml"/>
</mappers>
# mapper標簽,通過 resource 屬性引入 classpath 路徑的相對資源
3、類注冊引入
<mappers>
<mapper class="com.fyy.mapper.FlowerMapper"/>
</mappers>
# mapper 標簽,通過 class 屬性指定 mapper 介面名稱,此時對應的映射檔案必須與介面位于同一路徑
下,并且名稱相同
4、使用 url 絕對路徑方式引入(不推薦)
<mappers>
<mapper url="file:///var/mappers/UserMapper.xml"/>
</mappers>
# mapper 標簽,通過 url 引入網路資源或者本地磁盤資源
總結
只有配置了 mappers 資訊,Mybatis 才知道去哪里加載 Mapper 組態檔,開發中,根據專案中 Mapper 的配置偏好,選擇整合組態檔的配置方式,
2 Spring 整合 Mybatis
2.1 Spring 整合 dao 層
<?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/beans
https://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">
<!-- 配置整合Mybatis-->
<!--1、關聯資料庫組態檔-->
<context:property-placeholder location="classpath:database.properties"/>
<!--2、連接池-->
<!--資料庫連接池
dbcp 半自動化操作 不能自動連接
c3p0 自動化操作(自動的加載組態檔 并且設定到物件里面)-->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<!-- 配置連接池屬性 -->
<property name="driverClass" value="${jdbc.driver}"/>
<property name="jdbcUrl" value="${jdbc.url}"/>
<property name="user" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
<!-- c3p0連接池的私有屬性 -->
<property name="maxPoolSize" value="30"/>
<property name="minPoolSize" value="10"/>
<!-- 關閉連接后不自動commit -->
<property name="autoCommitOnClose" value="false"/>
<!-- 獲取連接超時時間 -->
<property name="checkoutTimeout" value="10000"/>
<!-- 當獲取連接失敗重試次數 -->
<property name="acquireRetryAttempts" value="2"/>
</bean>
<!-- 3.配置SqlSessionFactory物件 -->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<!-- 注入資料庫連接池 -->
<property name="dataSource" ref="dataSource"/>
<!-- 配置MyBaties全域組態檔:mybatis-config.xml -->
<property name="configLocation" value="classpath:mybatis-config.xml"/>
</bean>
<!-- 4.配置掃描Dao介面包,動態實作Dao介面注入到spring容器中 -->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
<!-- 注入sqlSessionFactory -->
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/>
<!-- 給出需要掃描Dao介面包 -->
<property name="basePackage" value="com.fyy.dao"/>
</bean>
</beans>
2.1.1 context:property-placeholder標簽作用?
在開發中有些引數是常量,比如:連接資料庫的 url,password,username 等資訊,不需要常變化,但在不同階段又需要變化,有沒有有一種方案方便我們在一個階段內不需要頻繁寫一個引數的值,但在不同階段間又可以方便的切換引數的配置資訊,
解決方案:Spring 3 中提供了一種簡便的方式就是 content:property-placeholder元素,即可解決上述問題,
在 content:property-placeholder 標簽中 利用屬性 localtion ,值為檔案路徑,可以使用 classpath: 指定檔案名即可,這里我們關聯上資料庫配置資訊,
Spring 容器中僅允許最多定義一個 context:property-placeholder ,其余的會被忽略,
2.1.2 資料庫連接池
每次創建一個資料庫連接都是一次極大的資源消費,所以連接復用,通過建立一個資料庫連接池以及一套連接使用管理策略,可以使一個資料庫連接可以得到高效、安全的復用,能避免了資料庫連接池頻繁建立、關閉的開銷,
連接池的本質:介于 Java 和 JDBC 之間的 Java jar 包!
第1步中,我們關聯了資料庫配置資訊,利用 bean 標簽可獲取其中的配置資訊,
可根據專案實際開發中,配置連接池的最大數量、最小數量、關閉連接后是否自動 commit、獲取連接超時時間、獲取連接失敗重試次數等,很多屬性,在這不一 一贅述了,
2.1.3 配置 SqlSessionFactory 物件
SqlSessionFacoty 的作用就是創建 SqlSession,SqlSession 就是一個會話,相當于 JDBC 中的 Connection 物件,每次應用訪問資料庫,就要通過 SqlSessionFactory 創建 SqlSession ,所以 SqlSessionFacoty 應該在 Mybatis 整個生命周期中,每一個資料庫只應該對應一個 SqlSessionFacoty.SqlSessionFacoty 在整個 Mybatis 運行流程中的作用如下圖,

配置 SqlSessionFactory 物件時,將資料庫連接池注入到其中,再配置 mybatis 全域組態檔 ,
我的理解是:在 SqlSessionFactory 創建 SqlSession 連接資料庫時,需要知道資料庫連接池的資訊,以及資料源,
2.1.4 配置掃描dao介面包
MapperScannerConfigurer 自動掃描將 Mapper 介面生成代理注入到 Spring容器,Mybatis 在與 Spring 整合的時候配置 MapperFactoryBean 來生成 Mapper 介面的代理, basePackage 屬性是讓你為映射器介面檔案設定基本的包路徑,你可以使用分號或者逗號作為分隔符設定多余一個包的路徑,每個映射器將會在指定的包路徑中遞回的被搜索到, 注意,沒有必要去指定 SqlSessionFactory 或 SqlSessionTemplate,因為 MapperScannerConfigurer 將會創建 MapperFactoryBean,之后自動裝配,但如果使用了一個以上的 DataSource,那么自動裝配可能會失效,這種情況下,可以使用 sqlSessionFactoryBeanName 或 sqlSessionTemplateBeanName 屬性來設定正確的 bean 名稱來使用,大白話理解:掃描dao層的介面,MapperScannerConfigurer 會自動為每個介面生成一個 介面代理,這個介面代理是介面的實作類,并注入到Spring容器中,
2.2 Spring 整合 service 層
<?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/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
<!-- 掃描service相關的bean -->
<context:component-scan base-package="com.fyy.service" />
<!--BookServiceImpl注入到IOC容器中-->
<bean id="BookServiceImpl" class="com.fyy.service.BookServiceImpl">
<property name="bookMapper" ref="bookMapper"/>
</bean>
<!-- 配置事務管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<!-- 注入資料庫連接池 -->
<property name="dataSource" ref="dataSource" />
</bean>
</beans>
事務管理器:DataSourceTransactionManager
事務管理是應用系統開發中必不可少的一部分,Spring 為事務管理提供了豐富的功能支持,Spring 事務管理分為編碼式和宣告式的兩種方式,編程式事務指的是通過編碼方式實作事務;宣告式事務基于 AOP,將具體業務邏輯與事務處理解耦,宣告式事務管理使業務代碼邏輯不受污染, 因此在實際使用中宣告式事務用的比較多,宣告式事務有兩種方式,一種是在組態檔(xml)中做相關的事務規則宣告,另一種是基于 @Transactional 注解的方式,
其中 組態檔宣告事務,是將資料庫連接池注入到 事務管理器中,關于注解方式宣告事務,就不在這里做過多介紹,后期會單獨拿出來分享Spring事務,
3 整合 SpringMVC
web.xml 組態檔

3.1 web.xml 檔案是如何執行的?
3.1.1 執行順序
訪問順序為:1->2->3->4,其中 2 和 3 的值必須相同,url-pattern標簽代表當一個請求發送到 servlet 容器時,容器會先將請求的 url 減去 當前應用背景關系的路徑作為 servlet 的映射 url,比如我訪問的是 http://localhost/test/aaa.html,我的應用背景關系是 test,容器會將 http://localhost/test 去掉,剩下的 /aaa.html 部分拿來做 servlet 映射匹配,如果與 設定的 url-pattern 映射匹配成功,請求才會被 DispatcherServlet 處理,圖中的 <url-pattern> 中的值為:/,代表所有請求都會被處理,
而被DispatcherServlet 處理前,還會經過 2 ,3 步驟,url 匹配成功后,這個 url 訪問名為 servlet-name 中值的 servlet,兩個 servlet-name 值必須相同,因為 servlet 標簽中的 servlet-name 標簽映射到 servlet-class 標簽中的值,最終訪問 servlet-class 標簽中的 DispatcherServlet 類,此時,請求才會被 DispatcherServlet 處理,
3.1.2 init-param標簽
圖中 \標簽中的 \標簽作用是,設定 springmvc 組態檔位置以及名稱,springmvc組態檔不設定默認位置是:webapp,可以使用 classpath 設定檔案的名稱為:spring-mvc.xml ,代表 springmvc 組態檔名必須為 spring-mvc.xml,如不使用 classpath 設定其路徑及名稱,默認在 webapp下,名稱為:\標簽中的值 + "-servlet.xml",例如:如 標簽中值為:springmvc,則默認的 springmvc 組態檔名為:springmvc-servlet.xml3.1.3 load-on-startup標簽
圖中 \標簽作用是,設定 servlet加載時間,如不設定默認在第一次請求訪問時加載 servlet,若設定此標簽值為正整數,會將 servlet 的加載時間提前到專案啟動時,此標簽中可以寫整數,但寫負整數和0和沒有設定是一樣的效果,只有設定為正整數才會將 servlet 的加載時間提前到專案啟動時,也就是 tomcat 啟動時,值越小,代表優先級越高,3.2 springmvc 組態檔是如何執行的?
spring-mvc.xml 組態檔

3.2.1 mvc:annotation-driven作用?
springmvc注解驅動會自動注冊:DefaultAnnotationHandlerMapping 與 AnnotationMethodHandlerAdapter 兩個bean,這兩個 bean 是 SpringMVC 為 @Controller 分發請求所必須的,解決了使用 @Controller 注解的前提配置,在 Spring MVC 3.1 以上:
- DefaultAnnotationHandlerMapping 變更為:RequestMappingHandlerMapping
- AnnotationMethodHandlerAdapter 變更為:RequestMappingHandlerAdapter
我們來看看這個類主要做了什么作業?
AnnotationDrivenBeanDefinitionParser類向工廠中注冊了幾個 bean 實體,其中包括:
- RequestMappingHandlerMapping
- BeanNameUrlHandlerMapping
- RequestMappingHandlerAdapter
- HttpRequestHandlerAdapter
- SimpleControllerHandlerAdapter
- ExceptionHandlerExceptionResolver
- ResponseStatusExceptionResolver
- DefaultHandlerExceptionResolver
上面的bean實體都是做什么的呢?
前兩個 bean 是 HandlerMapping 介面的實作類,用來處理請求映射的,其中第一個處理是 @RequestMapping 注解的,第二個是將 controller 類的名字映射為 請求 url,中間三個 adpter 是用來處理請求的,具體說就是確定呼叫哪個 controller 的哪個方法來處理當前請求,第一個處理 @Controller 注解的處理器,支持自定義方法引數和回傳值,第二個是處理繼承 HttpRequestHandler 的處理器,第三個處理繼承自 Controller 介面的處理器,后面三個是用來處理例外的決議器,
結論
如果使用@Controller 注解,沒有配置注解驅動: <mvc:annotation-driven> 的話,那么所有的請求都無法找到 DispatcherServlet ,并無法把請求分發至控制器,添加注解驅動后,才會掃描所有帶有 @Controller 注解的類,由 spring 管理并維護,
3.2.2 mvc:default-servlet-handler作用?
還記得我們在 web.xml 中配置了 url-pattern,用來過濾請求,SpringMVC 將接收到的所有請求都看做是一個普通請求,包括對于靜態資源的請求,這樣一來,所有對于靜態資源的請求都會被看做是一個普通的后臺控制器請求,而靜態資源的請求會因為找不到資源而報404例外,查看 tomcat 日志就可以看到會有警告,
對于此問題 SpringMVC 在全域組態檔中提供了一個 <mvc:default-servlet-handler>標簽,在 web 容器啟動的時候會在背景關系中定義一個 DefaultServletHttpRequestHandler ,它會對 DispatcherServlet 請求進行處理,如果該請求已經做了映射,那么會接著交給后臺對應的處理程式,如果沒有映射,就交給 web 應用服務器默認的 servlet 處理,從而找到對應的靜態資源,只有找不到靜態資源時會報錯,
如果默認的 Servlet 容器不用默認的 default ,用不同名稱進行自定義配置,或者在預設 Servlet 名稱未知的情況下使用了不同的 Servlet 容器,則必須顯示提供默認 Servlet 的名稱,
<mvc:default-servlet-handler default-servlet-name="myCustomDefaultServlet"/>
3.2.3 context:component-scan 作用?
掃描組件,將所有使用 @Controller 注解的類作為 SpringMVC 的控制層,其中 base-package 屬性是指定掃描的包,3.2.4 視圖決議器 作用?
視圖決議器,是將 prefix + 視圖名稱 + suffix = 確定最終要跳轉的頁面,其中視圖名稱是什么?? 處理請求的方法會回傳一個字串,這個字串即視圖名稱,最侄訓通過組態檔中配置的視圖決議器實作頁面的跳轉,例如:@Controller
public class TestController {
@RequestMapping("hello")
public String hello() {
System.out.println("success");
return "success";
}
}
# 其中回傳字串為:success,所以處理此請求最終要跳轉的頁面為:/WEB-INF/view/success.jsp
關于 SpringMVC的講解可以查看小編的另一篇文章,SpringMVC從基礎到原始碼
4 Spring 配置整合檔案
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
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">
<import resource="classpath:spirng/spring-dao.xml"/>
<import resource="classpath:spirng/spring-service.xml"/>
<import resource="classpath:spirng/spring-mvc.xml"/>
</beans>
applicationContext.xml,Spring 的核心組態檔,也可以通過組態檔修改名稱,
什么時候讀取組態檔,當我們在客戶端新建 ApplicationContext 實體時,會加載此組態檔,其中常用的 ApplicationContext 實作類有:
- ClassPathXmlApplicationContext
- ClassPathResource
- XmlWebApplicationContext
- FileSystemXmlApplicationContext
- …
結束
以上就是本人學習ssm框架配置的總結,有的地方不夠深入,但對于初學者來說應該也會識訓頗豐,如有錯誤,還請大佬們指正!
轉載請註明出處,本文鏈接:https://www.uj5u.com/qita/83848.html
標籤:其他
