Spring是一個輕量級Java開發框架,最先有Rod Johnson建立,目的是爲了解決企業級應用開發的業務邏輯層和其餘各層的耦合問題。它是一個分層的JavaSE/JavaEE full-stack(一站式)輕量級開源框架,爲開發Java應用程序提供全面的基礎架構支持。Spring負責基礎架構,所以Java開發者能夠專一於應用程序的開發。html
Spring最根本的使命是解決企業級應用開發的複雜性,即簡化Java開發。java
Spring能夠作不少事情,它爲企業級開發提供給了豐富的功能,可是這些功能的底層都依賴於它的兩個核心特性,也就是依賴注入(dependency injection,DI)*和*面向切面編程(aspect-oriented programming,AOP)。git
爲了下降Java開發的複雜性,Spring採起了如下4種關鍵策略github
Spring設計目標:Spring爲開發者提供一個一站式輕量級應用開發平臺;web
Spring設計理念:在JavaEE開發中,支持POJO和JavaBean開發方式,使應用面向接口開發,充分支持OO(面向對象)設計方法;Spring經過IoC容器實現對象耦合關係的管理,並實現依賴反轉,將對象之間的依賴關係交給IoC容器,實現解耦;正則表達式
Spring框架的核心:IoC容器和AOP模塊。經過IoC容器管理POJO對象以及他們之間的耦合關係;經過AOP以動態非侵入的方式加強服務。spring
IoC讓相互協做的組件保持鬆散的耦合,而AOP編程容許你把遍及於應用各層的功能分離出來造成可重用的功能組件。數據庫
優勢編程
方便解耦,簡化開發設計模式
Spring就是一個大工廠,能夠將全部對象的建立和依賴關係的維護,交給Spring管理。
AOP編程的支持
Spring提供面向切面編程,能夠方便的實現對程序進行權限攔截、運行監控等功能。
聲明式事務的支持
只須要經過配置就能夠完成對事務的管理,而無需手動編程。
方便程序的測試
Spring對Junit4支持,能夠經過註解方便的測試Spring程序。
方便集成各類優秀框架
Spring不排斥各類優秀的開源框架,其內部提供了對各類優秀框架的直接支持(如:Struts、Hibernate、MyBatis等)。
下降JavaEE API的使用難度
Spring對JavaEE開發中很是難用的一些API(JDBC、JavaMail、遠程調用等),都提供了封裝,使這些API應用難度大大下降。
缺點
應用場景:JavaEE企業應用開發,包括SSH、SSM等
Spring價值:
Spring 總共大約有 20 個模塊, 由 1300 多個不一樣的文件構成。 而這些組件被分別整合在核心容器(Core Container)
、 AOP(Aspect Oriented Programming)和設備支持(Instrmentation)
、數據訪問與集成(Data Access/Integeration)
、 Web
、 消息(Messaging)
、 Test
等 6 個模塊中。 如下是 Spring 5 的模塊結構圖:
這是基本的Spring模塊,提供spring 框架的基礎功能,BeanFactory 是 任何以spring爲基礎的應用的核心。Spring 框架創建在此模塊之上,它使Spring成爲一個容器。
Bean 工廠是工廠模式的一個實現,提供了控制反轉功能,用來把應用的配置和依賴從真正的應用代碼中分離。最經常使用的就是org.springframework.beans.factory.xml.XmlBeanFactory ,它根據XML文件中的定義加載beans。該容器從XML 文件讀取配置元數據並用它去建立一個徹底配置的系統或應用。
Spring 提供瞭如下5種標準的事件:
Spring 應用通常有如下組件:
使用 Spring 有如下方式:
控制反轉即IoC (Inversion of Control),它把傳統上由程序代碼直接操控的對象的調用權交給容器,經過容器來實現對象組件的裝配和管理。所謂的「控制反轉」概念就是對組件對象控制權的轉移,從程序代碼自己轉移到了外部容器。
Spring IOC 負責建立對象,管理對象(經過依賴注入(DI),裝配對象,配置對象,而且管理這些對象的整個生命週期。
Spring 中的 IoC 的實現原理就是工廠模式加反射機制。
示例:
interface Fruit { public abstract void eat(); } class Apple implements Fruit { public void eat(){ System.out.println("Apple"); } } class Orange implements Fruit { public void eat(){ System.out.println("Orange"); } } class Factory { public static Fruit getInstance(String ClassName) { Fruit f=null; try { f=(Fruit)Class.forName(ClassName).newInstance(); } catch (Exception e) { e.printStackTrace(); } return f; } } class Client { public static void main(String[] a) { Fruit f=Factory.getInstance("io.github.dunwu.spring.Apple"); if(f!=null){ f.eat(); } } }
Spring 的 IoC 設計支持如下功能:
其中,最重要的就是依賴注入,從 XML 的配置上說,即 ref 標籤。對應 Spring RuntimeBeanReference 對象。
對於 IoC 來講,最重要的就是容器。容器管理着 Bean 的生命週期,控制着 Bean 的依賴注入。
BeanFactory和ApplicationContext是Spring的兩大核心接口,均可以當作Spring的容器。其中ApplicationContext是BeanFactory的子接口。
依賴關係
BeanFactory:是Spring裏面最底層的接口,包含了各類Bean的定義,讀取bean配置文檔,管理bean的加載、實例化,控制bean的生命週期,維護bean之間的依賴關係。
ApplicationContext接口做爲BeanFactory的派生,除了提供BeanFactory所具備的功能外,還提供了更完整的框架功能:
加載方式
BeanFactroy採用的是延遲加載形式來注入Bean的,即只有在使用到某個Bean時(調用getBean()),纔對該Bean進行加載實例化。這樣,咱們就不能發現一些存在的Spring的配置問題。若是Bean的某一個屬性沒有注入,BeanFacotry加載後,直至第一次使用調用getBean方法纔會拋出異常。
ApplicationContext,它是在容器啓動時,一次性建立了全部的Bean。這樣,在容器啓動時,咱們就能夠發現Spring中存在的配置錯誤,這樣有利於檢查所依賴屬性是否注入。 ApplicationContext啓動後預載入全部的單實例Bean,經過預載入單實例bean ,確保當你須要的時候,你就不用等待,由於它們已經建立好了。
相對於基本的BeanFactory,ApplicationContext 惟一的不足是佔用內存空間。當應用程序配置Bean較多時,程序啓動較慢。
建立方式
BeanFactory一般以編程的方式被建立,ApplicationContext還能以聲明的方式建立,如使用ContextLoader。
註冊方式
BeanFactory和ApplicationContext都支持BeanPostProcessor、BeanFactoryPostProcessor的使用,但二者之間的區別是:BeanFactory須要手動註冊,而ApplicationContext則是自動註冊。
Spring 做者 Rod Johnson 設計了兩個接口用以表示容器。
BeanFactory 簡單粗暴,能夠理解爲就是個 HashMap,Key 是 BeanName,Value 是 Bean 實例。一般只提供註冊(put),獲取(get)這兩個功能。咱們能夠稱之爲 「低級容器」。
ApplicationContext 能夠稱之爲 「高級容器」。由於他比 BeanFactory 多了更多的功能。他繼承了多個接口。所以具有了更多的功能。例如資源的獲取,支持多種消息(例如 JSP tag 的支持),對 BeanFactory 多了工具級別的支持等待。因此你看他的名字,已經不是 BeanFactory 之類的工廠了,而是 「應用上下文」, 表明着整個大容器的全部功能。該接口定義了一個 refresh 方法,此方法是全部閱讀 Spring 源碼的人的最熟悉的方法,用於刷新整個容器,即從新加載/刷新全部的 bean。
固然,除了這兩個大接口,還有其餘的輔助接口,這裏就不介紹他們了。
BeanFactory和ApplicationContext的關係
爲了更直觀的展現 「低級容器」 和 「高級容器」 的關係,這裏經過經常使用的 ClassPathXmlApplicationContext 類來展現整個容器的層級 UML 關係。
有點複雜? 先不要慌,我來解釋一下。
最上面的是 BeanFactory,下面的 3 個綠色的,都是功能擴展接口,這裏就不展開講。
看下面的隸屬 ApplicationContext 粉紅色的 「高級容器」,依賴着 「低級容器」,這裏說的是依賴,不是繼承哦。他依賴着 「低級容器」 的 getBean 功能。而高級容器有更多的功能:支持不一樣的信息源頭,能夠訪問文件資源,支持應用事件(Observer 模式)。
一般用戶看到的就是 「高級容器」。 但 BeanFactory 也很是夠用啦!
左邊灰色區域的是 「低級容器」, 只負載加載 Bean,獲取 Bean。容器其餘的高級功能是沒有的。例如上圖畫的 refresh 刷新 Bean 工廠全部配置,生命週期事件回調等。
小結
說了這麼多,不知道你有沒有理解Spring IoC? 這裏小結一下:IoC 在 Spring 裏,只須要低級容器就能夠實現,2 個步驟:
上面就是 Spring 低級容器(BeanFactory)的 IoC。
至於高級容器 ApplicationContext,他包含了低級容器的功能,當他執行 refresh 模板方法的時候,將刷新整個容器的 Bean。同時其做爲高級容器,包含了太多的功能。一句話,他不只僅是 IoC。他支持不一樣信息源頭,支持 BeanFactory 工具類,支持層級容器,支持訪問文件資源,支持事件發佈通知,支持接口回調等等。
FileSystemXmlApplicationContext :此容器從一個XML文件中加載beans的定義,XML Bean 配置文件的全路徑名必須提供給它的構造函數。
ClassPathXmlApplicationContext:此容器也從一個XML文件中加載beans的定義,這裏,你須要正確設置classpath由於這個容器將在classpath裏找bean配置。
WebXmlApplicationContext:此容器加載一個XML文件,此文件定義了一個WEB應用的全部bean。
控制反轉IoC是一個很大的概念,能夠用不一樣的方式來實現。其主要實現方式有兩種:依賴注入和依賴查找
依賴注入:相對於IoC而言,依賴注入(DI)更加準確地描述了IoC的設計理念。所謂依賴注入(Dependency Injection),即組件之間的依賴關係由容器在應用系統運行期來決定,也就是由容器動態地將某種依賴關係的目標對象實例注入到應用系統中的各個關聯的組件之中。組件不作定位查詢,只提供普通的Java方法讓容器去決定依賴關係。
依賴注入的基本原則是:應用組件不該該負責查找資源或者其餘依賴的協做對象。配置對象的工做應該由IoC容器負責,「查找資源」的邏輯應該從應用組件的代碼中抽取出來,交給IoC容器負責。容器全權負責組件的裝配,它會把符合依賴關係的對象經過屬性(JavaBean中的setter)或者是構造器傳遞給須要的對象。
依賴注入之因此更流行是由於它是一種更可取的方式:讓容器全權負責依賴查詢,受管組件只須要暴露JavaBean的setter方法或者帶參數的構造器或者接口,使容器能夠在初始化時組裝對象的依賴關係。其與依賴查找方式相比,主要優點爲:
依賴注入是時下最流行的IoC實現方式,依賴注入分爲接口注入(Interface Injection),Setter方法注入(Setter Injection)和構造器注入(Constructor Injection)三種方式。其中接口注入因爲在靈活性和易用性比較差,如今從Spring4開始已被廢棄。
構造器依賴注入:構造器依賴注入經過容器觸發一個類的構造器來實現的,該類有一系列參數,每一個參數表明一個對其餘類的依賴。
Setter方法注入:Setter方法注入是容器經過調用無參構造器或無參static工廠 方法實例化bean以後,調用該bean的setter方法,即實現了基於setter的依賴注入。
兩種依賴方式均可以使用,構造器注入和Setter方法注入。最好的解決方案是用構造器參數實現強制依賴,setter方法實現可選依賴。
Spring beans 是那些造成Spring應用的主幹的java對象。它們被Spring IOC容器初始化,裝配,和管理。這些beans經過容器中配置的元數據建立。好比,以XML文件中 的形式定義。
一個Spring Bean 的定義包含容器必知的全部配置元數據,包括如何建立一個bean,它的生命週期詳情及它的依賴。
這裏有三種重要的方法給Spring 容器提供配置元數據。
Spring配置文件是個XML 文件,這個文件包含了類信息,描述瞭如何配置它們,以及如何相互調用。
當定義一個 在Spring裏,咱們還能給這個bean聲明一個做用域。它能夠經過bean 定義中的scope屬性來定義。如,當Spring要在須要的時候每次生產一個新的bean實例,bean的scope屬性被指定爲prototype。另外一方面,一個bean每次使用的時候必須返回同一個實例,這個bean的scope 屬性 必須設爲 singleton。
Spring框架支持如下五種bean的做用域:
注意: 缺省的Spring bean 的做用域是Singleton。使用 prototype 做用域須要慎重的思考,由於頻繁建立和銷燬 bean 會帶來很大的性能開銷。
不是,Spring框架中的單例bean不是線程安全的。
spring 中的 bean 默認是單例模式,spring 框架並無對單例 bean 進行多線程的封裝處理。
實際上大部分時候 spring bean 無狀態的(好比 dao 類),全部某種程度上來講 bean 也是安全的,但若是 bean 有狀態的話(好比 view model 對象),那就要開發者本身去保證線程安全了,最簡單的就是改變 bean 的做用域,把「singleton」變動爲「prototype」,這樣請求 bean 至關於 new Bean()了,因此就能夠保證線程安全了。
在通常狀況下,只有無狀態的Bean才能夠在多線程環境下共享,在Spring中,絕大部分Bean均可以聲明爲singleton做用域,由於Spring對一些Bean中非線程安全狀態採用ThreadLocal進行處理,解決線程安全問題。
ThreadLocal和線程同步機制都是爲了解決多線程中相同變量的訪問衝突問題。同步機制採用了「時間換空間」的方式,僅提供一份變量,不一樣的線程在訪問前須要獲取鎖,沒得到鎖的線程則須要排隊。而ThreadLocal採用了「空間換時間」的方式。
ThreadLocal會爲每個線程提供一個獨立的變量副本,從而隔離了多個線程對數據的訪問衝突。由於每個線程都擁有本身的變量副本,從而也就沒有必要對該變量進行同步了。ThreadLocal提供了線程安全的共享對象,在編寫多線程代碼時,能夠把不安全的變量封裝進ThreadLocal。
在傳統的Java應用中,bean的生命週期很簡單。使用Java關鍵字new進行bean實例化,而後該bean就可使用了。一旦該bean再也不被使用,則由Java自動進行垃圾回收。相比之下,Spring容器中的bean的生命週期就顯得相對複雜多了。正確理解Spring bean的生命週期很是重要,由於你或許要利用Spring提供的擴展點來自定義bean的建立過程。下圖展現了bean裝載到Spring應用上下文中的一個典型的生命週期過程。
bean在Spring容器中從建立到銷燬經歷了若干階段,每一階段均可以針對Spring如何管理bean進行個性化定製。
正如你所見,在bean準備就緒以前,bean工廠執行了若干啓動步驟。
咱們對上圖進行詳細描述:
Spring對bean進行實例化;
Spring將值和bean的引用注入到bean對應的屬性中;
若是bean實現了BeanNameAware接口,Spring將bean的ID傳遞給setBean-Name()方法;
若是bean實現了BeanFactoryAware接口,Spring將調用setBeanFactory()方法,將BeanFactory容器實例傳入;
若是bean實現了ApplicationContextAware接口,Spring將調用setApplicationContext()方法,將bean所在的應用上下文的引用傳入進來;
若是bean實現了BeanPostProcessor接口,Spring將調用它們的post-ProcessBeforeInitialization()方法;
若是bean實現了InitializingBean接口,Spring將調用它們的after-PropertiesSet()方法。相似地,若是bean使用initmethod聲明瞭初始化方法,該方法也會被調用;
若是bean實現了BeanPostProcessor接口,Spring將調用它們的post-ProcessAfterInitialization()方法;
此時,bean已經準備就緒,能夠被應用程序使用了,它們將一直駐留在應用上下文中,直到該應用上下文被銷燬;
若是bean實現了DisposableBean接口,Spring將調用它的destroy()接口方法。一樣,若是bean使用destroy-method聲明瞭銷燬方法,該方法也會被調用。
如今你已經瞭解瞭如何建立和加載一個Spring容器。可是一個空的容器並無太大的價值,在你把東西放進去以前,它裏面什麼都沒有。爲了從Spring的DI(依賴注入)中受益,咱們必須將應用對象裝配進Spring容器中。
有兩個重要的bean 生命週期方法,第一個是setup , 它是在容器加載bean的時候被調用。第二個方法是 teardown 它是在容器卸載類的時候被調用。
bean 標籤有兩個重要的屬性(init-method和destroy-method)。用它們你能夠本身定製初始化和註銷方法。它們也有相應的註解(@PostConstruct和@PreDestroy)。
在Spring框架中,當一個bean僅被用做另外一個bean的屬性時,它能被聲明爲一個內部bean。內部bean能夠用setter注入「屬性」和構造方法注入「構造參數」的方式來實現,內部bean一般是匿名的,它們的Scope通常是prototype。
Spring提供如下幾種集合的配置元素:
類型用於注入一列值,容許有相同的值。
類型用於注入一組值,不容許有相同的值。
類型用於注入一組鍵值對,鍵和值均可覺得任意類型。
類型用於注入一組鍵值對,鍵和值都只能爲String類型。
裝配,或bean 裝配是指在Spring 容器中把bean組裝到一塊兒,前提是容器須要知道bean的依賴關係,如何經過依賴注入來把它們裝配到一塊兒。
在Spring框架中,在配置文件中設定bean的依賴關係是一個很好的機制,Spring 容器可以自動裝配相互合做的bean,這意味着容器不須要和配置,能經過Bean工廠自動處理bean之間的協做。這意味着 Spring能夠經過向Bean Factory中注入的方式自動搞定bean之間的依賴關係。自動裝配能夠設置在每一個bean上,也能夠設定在特定的bean上。
在spring中,對象無需本身查找或建立與其關聯的其餘對象,由容器負責把須要相互協做的對象引用賦予各個對象,使用autowire來配置自動裝載模式。
在Spring框架xml配置中共有5種自動裝配:
使用@Autowired註解來自動裝配指定的bean。在使用@Autowired註解以前須要在Spring配置文件進行配置,<context:annotation-config />。
在啓動spring IoC時,容器自動裝載了一個AutowiredAnnotationBeanPostProcessor後置處理器,當容器掃描到@Autowied、@Resource或@Inject時,就會在IoC容器自動查找須要的bean,並裝配給該對象的屬性。在使用@Autowired時,首先在容器中查詢對應類型的bean:
自動裝配的侷限性是:
重寫:你仍需用 和 配置來定義依賴,意味着總要重寫自動裝配。
基本數據類型:你不能自動裝配簡單的屬性,如基本數據類型,String字符串,和類。
模糊特性:自動裝配不如顯式裝配精確,若是有可能,建議使用顯式裝配。
能夠。
基於Java的配置,容許你在少許的Java註解的幫助下,進行你的大部分Spring配置而非經過XML文件。
以@Configuration 註解爲例,它用來標記類能夠當作一個bean的定義,被Spring IOC容器使用。
另外一個例子是@Bean註解,它表示此方法將要返回一個對象,做爲一個bean註冊進Spring應用上下文。
@Configuration public class StudentConfig { @Bean public StudentBean myStudent() { return new StudentBean(); } }
註解裝配在默認狀況下是不開啓的,爲了使用註解裝配,咱們必須在Spring配置文件中配置 <context:annotation-config/>
元素。
@Component:這將 java 類標記爲 bean。它是任何 Spring 管理組件的通用構造型。spring 的組件掃描機制如今能夠將其拾取並將其拉入應用程序環境中。
@Controller:這將一個類標記爲 Spring Web MVC 控制器。標有它的 Bean 會自動導入到 IoC 容器中。
@Service:此註解是組件註解的特化。它不會對 @Component 註解提供任何其餘行爲。您能夠在服務層類中使用 @Service 而不是 @Component,由於它以更好的方式指定了意圖。
@Repository:這個註解是具備相似用途和功能的 @Component 註解的特化。它爲 DAO 提供了額外的好處。它將 DAO 導入 IoC 容器,並使未經檢查的異常有資格轉換爲 Spring DataAccessException。
這個註解代表bean的屬性必須在配置的時候設置,經過一個bean定義的顯式的屬性值或經過自動裝配,若@Required註解的bean屬性未被設置,容器將拋出BeanInitializationException。示例:
public class Employee { private String name; @Required public void setName(String name){ this.name=name; } public string getName(){ return name; } }
@Autowired默認是按照類型裝配注入的,默認狀況下它要求依賴對象必須存在(能夠設置它required屬性爲false)。@Autowired 註解提供了更細粒度的控制,包括在何處以及如何完成自動裝配。它的用法和@Required同樣,修飾setter方法、構造器、屬性或者具備任意名稱和/或多個參數的PN方法。
public class Employee { private String name; @Autowired public void setName(String name) { this.name=name; } public string getName(){ return name; } }
@Autowired可用於:構造函數、成員變量、Setter方法
@Autowired和@Resource之間的區別
當您建立多個相同類型的 bean 並但願僅使用屬性裝配其中一個 bean 時,您可使用@Qualifier 註解和 @Autowired 經過指定應該裝配哪一個確切的 bean 來消除歧義。
@RequestMapping 註解用於將特定 HTTP 請求方法映射到將處理相應請求的控制器中的特定類/方法。此註釋可應用於兩個級別:
Spring 經過提供ORM模塊,支持咱們在直接JDBC之上使用一個對象/關係映射映射(ORM)工具,Spring 支持集成主流的ORM框架,如Hiberate,JDO和 iBATIS,JPA,TopLink,JDO,OJB 。Spring的事務管理一樣支持以上全部ORM框架及JDBC。
使用Spring JDBC 框架,資源管理和錯誤處理的代價都會被減輕。因此開發者只需寫statements 和 queries從數據存取數據,JDBC也能夠在Spring框架提供的模板類的幫助下更有效地被使用,這個模板叫JdbcTemplate
經過使用JDBC抽象和DAO模塊,保證數據庫代碼的簡潔,並能避免數據庫資源錯誤關閉致使的問題,它在各類不一樣的數據庫的錯誤信息之上,提供了一個統一的異常訪問層。它還利用Spring的AOP 模塊給Spring應用中的對象提供事務管理服務。
Spring DAO(數據訪問對象) 使得 JDBC,Hibernate 或 JDO 這樣的數據訪問技術更容易以一種統一的方式工做。這使得用戶容易在持久性技術之間切換。它還容許您在編寫代碼時,無需考慮捕獲每種技術不一樣的異常。
JdbcTemplate
SimpleJdbcTemplate
NamedParameterJdbcTemplate
SimpleJdbcInsert
SimpleJdbcCall
JdbcTemplate 類提供了不少便利的方法解決諸如把數據庫數據轉變成基本數據類型或對象,執行寫好的或可調用的數據庫操做語句,提供自定義的數據錯誤處理。
在Spring中有兩種方式訪問Hibernate:
用Spring的 SessionFactory 調用 LocalSessionFactory。集成過程分三步:
Spring支持兩種類型的事務管理:
編程式事務管理:這意味你經過編程的方式管理事務,給你帶來極大的靈活性,可是難維護。
聲明式事務管理:這意味着你能夠將業務代碼和事務管理分離,你只需用註解和XML配置來管理事務。
Spring事務的本質其實就是數據庫對事務的支持,沒有數據庫的事務支持,spring是沒法提供事務功能的。真正的數據庫層的事務提交和回滾是經過binlog或者redo log實現的。
spring事務的傳播行爲說的是,當多個事務同時存在的時候,spring如何處理這些事務的行爲。
① PROPAGATION_REQUIRED:若是當前沒有事務,就建立一個新事務,若是當前存在事務,就加入該事務,該設置是最經常使用的設置。 ② PROPAGATION_SUPPORTS:支持當前事務,若是當前存在事務,就加入該事務,若是當前不存在事務,就以非事務執行。 ③ PROPAGATION_MANDATORY:支持當前事務,若是當前存在事務,就加入該事務,若是當前不存在事務,就拋出異常。 ④ PROPAGATION_REQUIRES_NEW:建立新事務,不管當前存不存在事務,都建立新事務。 ⑤ PROPAGATION_NOT_SUPPORTED:以非事務方式執行操做,若是當前存在事務,就把當前事務掛起。 ⑥ PROPAGATION_NEVER:以非事務方式執行,若是當前存在事務,則拋出異常。 ⑦ PROPAGATION_NESTED:若是當前存在事務,則在嵌套事務內執行。若是當前沒有事務,則按REQUIRED屬性執行。
spring 有五大隔離級別,默認值爲 ISOLATION_DEFAULT(使用數據庫的設置),其餘四個隔離級別和數據庫的隔離級別一致:
髒讀 :表示一個事務可以讀取另外一個事務中還未提交的數據。好比,某個事務嘗試插入記錄 A,此時該事務還未提交,而後另外一個事務嘗試讀取到了記錄 A。
不可重複讀 :是指在一個事務內,屢次讀同一數據。
幻讀 :指同一個事務內屢次查詢返回的結果集不同。好比同一個事務 A 第一次查詢時候有 n 條記錄,可是第二次同等條件下查詢卻有 n+1 條記錄,這就好像產生了幻覺。發生幻讀的緣由也是另一個事務新增或者刪除或者修改了第一個事務結果集裏面的數據,同一個記錄的數據內容被修改了,全部數據行的記錄就變多或者變少了。
大多數Spring框架的用戶選擇聲明式事務管理,由於它對應用代碼的影響最小,所以更符合一個無侵入的輕量級容器的思想。聲明式事務管理要優於編程式事務管理,雖然比編程式事務管理(這種方式容許你經過代碼控制事務)少了一點靈活性。惟一不足地方是,最細粒度只能做用到方法級別,沒法作到像編程式事務那樣能夠做用到代碼塊級別。
OOP(Object-Oriented Programming)面向對象編程,容許開發者定義縱向的關係,但並適用於定義橫向的關係,致使了大量代碼的重複,而不利於各個模塊的重用。
AOP(Aspect-Oriented Programming),通常稱爲面向切面編程,做爲面向對象的一種補充,用於將那些與業務無關,但卻對多個對象產生影響的公共行爲和邏輯,抽取並封裝爲一個可重用的模塊,這個模塊被命名爲「切面」(Aspect),減小系統中的重複代碼,下降了模塊間的耦合度,同時提升了系統的可維護性。可用於權限認證、日誌、事務處理等。
AOP實現的關鍵在於 代理模式,AOP代理主要分爲靜態代理和動態代理。靜態代理的表明爲AspectJ;動態代理則以Spring AOP爲表明。
(1)AspectJ是靜態代理的加強,所謂靜態代理,就是AOP框架會在編譯階段生成AOP代理類,所以也稱爲編譯時加強,他會在編譯階段將AspectJ(切面)織入到Java字節碼中,運行的時候就是加強以後的AOP對象。
(2)Spring AOP使用的動態代理,所謂的動態代理就是說AOP框架不會去修改字節碼,而是每次運行時在內存中臨時爲方法生成一個AOP對象,這個AOP對象包含了目標對象的所有方法,而且在特定的切點作了加強處理,並回調原對象的方法。
Spring AOP中的動態代理主要有兩種方式,JDK動態代理和CGLIB動態代理:
靜態代理與動態代理區別在於生成AOP代理對象的時機不一樣,相對來講AspectJ的靜態代理方式具備更好的性能,可是AspectJ須要特定的編譯器進行處理,而Spring AOP則無需特定的編譯器處理。
InvocationHandler 的 invoke(Object proxy,Method method,Object[] args):proxy是最終生成的代理實例; method 是被代理目標實例的某個具體方法; args 是被代理目標實例某個方法的具體入參, 在方法反射調用時使用。
將 Advice 應用於目標對象後建立的對象稱爲代理。在客戶端對象的狀況下,目標對象和代理對象是相同的。
Advice + Target Object = Proxy
(1)切面(Aspect):切面是通知和切點的結合。通知和切點共同定義了切面的所有內容。 在Spring AOP中,切面可使用通用類(基於模式的風格) 或者在普通類中以 @AspectJ 註解來實現。
(2)鏈接點(Join point):指方法,在Spring AOP中,一個鏈接點 老是 表明一個方法的執行。 應用可能有數以千計的時機應用通知。這些時機被稱爲鏈接點。鏈接點是在應用執行過程當中可以插入切面的一個點。這個點能夠是調用方法時、拋出異常時、甚至修改一個字段時。切面代碼能夠利用這些點插入到應用的正常流程之中,並添加新的行爲。
(3)通知(Advice):在AOP術語中,切面的工做被稱爲通知。
(4)切入點(Pointcut):切點的定義會匹配通知所要織入的一個或多個鏈接點。咱們一般使用明確的類和方法名稱,或是利用正則表達式定義所匹配的類和方法名稱來指定這些切點。
(5)引入(Introduction):引入容許咱們向現有類添加新方法或屬性。
(6)目標對象(Target Object): 被一個或者多個切面(aspect)所通知(advise)的對象。它一般是一個代理對象。也有人把它叫作 被通知(adviced) 對象。 既然Spring AOP是經過運行時代理實現的,這個對象永遠是一個 被代理(proxied) 對象。
(7)織入(Weaving):織入是把切面應用到目標對象並建立新的代理對象的過程。在目標對象的生命週期裏有多少個點能夠進行織入:
經過在代理類中包裹切面,Spring在運行期把切面織入到Spring管理的bean中。代理封裝了目標類,並攔截被通知方法的調用,再把調用轉發給真正的目標bean。當代理攔截到方法調用時,在調用目標bean方法以前,會執行切面邏輯。
直到應用須要被代理的bean時,Spring才建立代理對象。若是使用的是ApplicationContext的話,在ApplicationContext從BeanFactory中加載全部bean的時候,Spring纔會建立被代理的對象。由於Spring運行時才建立代理對象,因此咱們不須要特殊的編譯器來織入SpringAOP的切面。
由於Spring基於動態代理,因此Spring只支持方法鏈接點。Spring缺乏對字段鏈接點的支持,並且它不支持構造器鏈接點。方法以外的鏈接點攔截功能,咱們能夠利用Aspect來補充。
關注點(concern)是應用中一個模塊的行爲,一個關注點可能會被定義成一個咱們想實現的一個功能。
橫切關注點(cross-cutting concern)是一個關注點,此關注點是整個應用都會使用的功能,並影響整個應用,好比日誌,安全和數據傳輸,幾乎應用的每一個模塊都須要的功能。所以這些都屬於橫切關注點。
在AOP術語中,切面的工做被稱爲通知,其實是程序執行時要經過SpringAOP框架觸發的代碼段。
Spring切面能夠應用5種類型的通知:
同一個aspect,不一樣advice的執行順序: ①沒有異常狀況下的執行順序: around before advice before advice target method 執行 around after advice after advice afterReturning ②有異常狀況下的執行順序: around before advice before advice target method 執行 around after advice after advice afterThrowing:異常發生 java.lang.RuntimeException: 異常發生
aspect 由 pointcount 和 advice 組成,切面是通知和切點的結合。 它既包含了橫切邏輯的定義, 也包括了鏈接點的定義. Spring AOP 就是負責實施切面的框架, 它將切面所定義的橫切邏輯編織到切面所指定的鏈接點中.
AOP 的工做重心在於如何將加強編織目標對象的鏈接點上, 這裏包含兩個工做:
能夠簡單地認爲, 使用 @Aspect 註解的類就是切面.
在這種狀況下,切面由常規類以及基於XML的配置實現。
在這種狀況下(基於@AspectJ的實現),涉及到的切面聲明的風格與帶有java5標註的普通java類一致。
BeanNameAutoProxyCreator
DefaultAdvisorAutoProxyCreator
Metadata autoproxying
做者:ThinkWon
來源: https://www.bianchengquan.com...