亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > Java > 正文

詳解Spring框架入門

2024-07-13 10:17:39
字體:
來源:轉載
供稿:網友

一、什么是Spring 

Spring框架是由于軟件開發的復雜性而創建的。Spring使用的是基本的JavaBean來完成以前只可能由EJB完成的事情。然而,Spring的用途不僅僅限于服務器端的開發。從簡單性、可測試性和松耦合性角度而言,絕大部分Java應用都可以從Spring中受益。Spring是一個輕量級控制反轉(IoC)和面向切面(AOP)的容器框架。
 ◆目的:解決企業應用開發的復雜性

 ◆功能:使用基本的JavaBean代替EJB,并提供了更多的企業應用功能

 ◆范圍:任何Java應用

二、什么是IOC

控制反轉(Inversion of Control,英文縮寫為IoC)把創建對象的權利交給框架,是框架的重要特征,并非面向對象編程的專用術語。它包括依賴注入和依賴查找。傳統的業務層,當需要資源時就在該業務層new資源,這樣耦合性(程序之間相互依賴關聯)較高?,F在將new的部分交給spring,做到高內聚低耦合。簡而言之:原先是每當調用dao層或service層方法時,由app來new,現在是將new的權利交給spring,要什么資源從spring中獲??!

三、快速搭建框架環境

1.下載框架所需的依賴jar包

 spring官網為:http://spring.io/

下載jar包:   http://repo.springsource.org/libs-release-local/org/springframework/spring

2.導入基本jar包

Spring,框架,Spring框架

其實基本核心jar有beans;context;core;expression包,其他是依賴log4j日志。當然spring的jar不止這些,后期慢慢加上。

3.配置log4j配置文件

日志文件定義在src目錄下

### direct log messages to stdout ###log4j.appender.stdout=org.apache.log4j.ConsoleAppenderlog4j.appender.stdout.Target=System.errlog4j.appender.stdout.layout=org.apache.log4j.PatternLayoutlog4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n### direct messages to file mylog.log ###log4j.appender.file=org.apache.log4j.FileAppenderlog4j.appender.file.File=c/:mylog.loglog4j.appender.file.layout=org.apache.log4j.PatternLayoutlog4j.appender.file.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n### set log levels - for more verbose logging change 'info' to 'debug' ###log4j.rootLogger=info, stdout

4.測試日志文件是否部署成功

package com.clj.demo1;import org.apache.log4j.Logger;import org.junit.Test;/** * 演示日志用法 * @author Administrator * */public class Demo1 { //創建日志類 private Logger log=Logger.getLogger(Demo1.class); @Test public void run1(){  //可以將log4j.rootLogger屬性中的info改為off則不會再控制臺顯示  log.info("執行了"); }}

5.定義一個接口和實現類

接口:

package com.clj.demo2;public interface UserService { public void sayHello();}

實現類

package com.clj.demo2;public class UserServiceImpl implements UserService{ private String name;  public String getName() {  return name; } public void setName(String name) {  this.name = name; } public void init(){  System.out.println("初始化。。"); } public void sayHello() {  System.out.println("Hello Spring"+"/t"+name); } public void destory(){  System.out.println("銷毀。。"); }}

6.定義spring專屬的配置文件

定義名為applicationContext.xml,位置為src下,與日志文件同目錄,導入相對應的約束,并將實現類注入到配置文件中,剛開始入門,使用bean約束

<?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:p="http://www.springframework.org/schema/p" xsi:schemaLocation="  http://www.springframework.org/schema/beans   http://www.springframework.org/schema/beans/spring-beans.xsd">  <!-- 使用bean標簽    1.id值唯一(必寫)  2.注意:class為實現類路徑,不是接口(必寫)  3.init-method核心方法執行之前初始化工作(選寫)  4.destroy-method核心方法執行之后初始化工作(選寫)-->  <bean id="userService" class="com.clj.demo2.UserServiceImpl" init-method="init" destroy-method="destory">   <property name="name" value="佳先森"></property>  </bean></beans>

7.測試

public class Demo1 { /**  * 原始方式  */ @Test public void run(){  //創建實現類  UserServiceImpl s=new UserServiceImpl();  s.setName("佳先森");  s.sayHello(); } /**  * 老的工廠版本BeanFactory  * 舊的工廠不會創建配置文件對象  */ @Test public void run2(){  BeanFactory factory=new XmlBeanFactory(new ClassPathResource("applicationContext.xml"));  UserService us=(UserService)factory.getBean("userService");  us.sayHello(); } /**  * 使用spring框架IOC方式  * 新版本factory創建啟動服務器會創建配置文件對象,再次調用時無需加載工廠  */ @Test public void run3(){  //創建工廠,加載核心配置文件(ClassPathXmlApplicationContext從src下找)  ApplicationContext ac=new ClassPathXmlApplicationContext("applicationContext.xml");  //從工廠中獲取到對象(配置文件中的id值,這里用了多態)  UserService usi=(UserService) ac.getBean("userService");  //調用對象的方法執行  usi.sayHello(); } /**  * 演示destroy-method方法  * bean摧毀方法不會自動執行  * 除非scope= singleton或者web容器中會自動調用,但是main函數或測試用例需要手動調用(需要使用ClassPathXmlApplicationContext的close()方法)  */ @Test public void run4(){  //創建工廠,加載核心配置文件(ClassPathXmlApplicationContext從src下找)  ClassPathXmlApplicationContext ac=new ClassPathXmlApplicationContext("applicationContext.xml");  //從工廠中獲取到對象(配置文件中的id值,這里用了多態)  UserService usi=(UserService) ac.getBean("userService");  //調用對象的方法執行  usi.sayHello();  //ApplicationContext實現類提供close方法,將工廠關閉就可執行destory-method方法  ac.close(); }}

其中舊工廠與新工廠的區別

* BeanFactory和ApplicationContext的區別

* BeanFactory               -- BeanFactory采取延遲加載,第一次getBean時才會初始化Bean

* ApplicationContext      -- 在加載applicationContext.xml時候就會創建具體的Bean對象的實例,還提供了一些其他的功能

 * 事件傳遞

 * Bean自動裝配

* 各種不同應用層的Context實現

總結:這是個最基本的demo,是將實現類配置到了spring配置文件中,每次啟動服務器時,就會加載配置文件,從而實例化了實現類

四、spring之依賴注入

1、什么是依賴注入?

Spring 能有效地組織J2EE應用各層的對象。不管是控制層的Action對象,還是業務層的Service對象,還是持久層的DAO對象,都可在Spring的 管理下有機地協調、運行。Spring將各層的對象以松耦合的方式組織在一起,Action對象無須關心Service對象的具體實現,Service對 象無須關心持久層對象的具體實現,各層對象的調用完全面向接口。當系統需要重構時,代碼的改寫量將大大減少。依賴注入讓bean與bean之間以配置文件組織在一起,而不是以硬編碼的方式耦合在一起。理解依賴注入

依賴注入(Dependency Injection)和控制反轉(Inversion of Control)是同一個概念。具體含義是:當某個角色(可能是一個Java實例,調用者)需要另一個角色(另一個Java實例,被調用者)的協助時,在 傳統的程序設計過程中,通常由調用者來創建被調用者的實例。但在Spring里,創建被調用者的工作不再由調用者來完成,因此稱為控制反轉;創建被調用者 實例的工作通常由Spring容器來完成,然后注入調用者,因此也稱為依賴注入。

不管是依賴注入,還是控制反轉,都說明Spring采用動態、靈活的方式來管理各種對象。對象與對象之間的具體實現互相透明。  

2. IOC和DI的概念

* IOC -- Inverse of Control,控制反轉,將對象的創建權反轉給Spring?。?/p>

* DI -- Dependency Injection,依賴注入,在Spring框架負責創建Bean對象時,動態的將依賴對象注入到Bean組件中?。?/p>

3.演示

對于類成員變量,常用的注入方式有兩種

屬性set方法注入和構造方法注入

先演示第一種:屬性set方法注入

1)持久層

package com.clj.demo3;public class CustomerDaoImpl {  public void save(){    System.out.println("我是持久層的Dao");  }}

2)業務層

注意:此時是想將持久層注入到業務層,將創建持久層實例權利交給框架,條件是業務層必須提供持久層的成員屬性和set方法

package com.clj.demo3;/** * 依賴注入之將dao 層注入到service層 * @author Administrator * */public class CustomerServiceImpl{  //提供成員屬相,提供set方法  private CustomerDaoImpl customerDao;    public void setCustomerDao(CustomerDaoImpl customerDao) {    this.customerDao = customerDao;  }  public void save(){    System.out.println("我是業務層的service...");    //1.原始方式    //new CustomerDaoImpl().save();        //2.spring 之IOC方式    customerDao.save();  }}

 3)配置文件配置

<!-- 演示依賴注入 -->   <bean id="customerDao" class="com.clj.demo3.CustomerDaoImpl"/>   <bean id="customerService" class="com.clj.demo3.CustomerServiceImpl">       <!-- 將Dao注入到service層 -->      <property name="customerDao" ref="customerDao"></property>   </bean>

4)測試

/**   * spring 依賴注入方式   * 將dao層注入到service層   */  @Test  public void run2(){    //創建工廠,加載配置文件,customerService被創建,從而也創建了customerDao    ApplicationContext context=new ClassPathXmlApplicationContext("applicationContext.xml");    CustomerServiceImpl csi=(CustomerServiceImpl) context.getBean("customerService");    csi.save();  }

第二種:構造方法注入

1)pojo類并提供構造方法

package com.clj.demo4;/** * 演示的構造方法的注入方式 * @author Administrator * */public class Car1 {  private String cname;  private Double price;  public Car1(String cname, Double price) {    super();    this.cname = cname;    this.price = price;  }  @Override  public String toString() {    return "Car1 [cname=" + cname + ", price=" + price + "]";  }  }

2)配置文件配置

 <!-- 演示構造方法注入方式 -->   <bean id="car1" class="com.clj.demo4.Car1">     <!-- 寫法一<constructor-arg name="cname" value="寶馬"/>     <constructor-arg name="price" value="400000"/> -->     <!--寫法二 -->     <constructor-arg index="0" value="寶馬"/>     <constructor-arg index="1" value="400000"/>    </bean>  

3)測試

@Test  public void run1(){    ApplicationContext ac=new ClassPathXmlApplicationContext("applicationContext.xml");    Car1 car=(Car1) ac.getBean("car1");    System.out.println(car);  }

拓展:構造方法之將一個對象注入到另一個對象中

1)pojo類:目的:將上列中的車注入到人類,使之成為其中一個屬性,則必須在此類中提供車的成員屬性,并提供有參構造方法

package com.clj.demo4;public class Person {  private String name;  private Car1 car1;  public Person(String name, Car1 car1) {    super();    this.name = name;    this.car1 = car1;  }  @Override  public String toString() {    return "Person [name=" + name + ", car1=" + car1 + "]";  }}

2)配置文件

 <!-- 構造方法之將一個對象注入到另一個對象-->   <bean id="person" class="com.clj.demo4.Person">     <constructor-arg name="name" value="佳先森"/>     <constructor-arg name="car1" ref="car1"/>   </bean>  

4.如何注入集合數組

1)定義pojo類

package com.clj.demo4;import java.util.Arrays;import java.util.List;import java.util.Map;import java.util.Properties;import java.util.Set;/** * 演示集合注入的方式 * @author Administrator * */public class User {  private String[] arrs;  private List<String> list;  private Set<String> sets;  private Map<String,String> map;  private Properties pro;    public void setPro(Properties pro) {    this.pro = pro;  }  public void setSets(Set<String> sets) {    this.sets = sets;  }  public void setMap(Map<String, String> map) {    this.map = map;  }  public void setList(List<String> list) {    this.list = list;  }  public void setArrs(String[] arrs) {    this.arrs = arrs;  }  @Override  public String toString() {    return "User [arrs=" + Arrays.toString(arrs) + ", list=" + list        + ", sets=" + sets + ", map=" + map + ", pro=" + pro + "]";  } }

2)配置文件

 <!-- 注入集合 -->   <bean id="user" class="com.clj.demo4.User">     <!-- 數組 -->     <property name="arrs">       <list>         <value>數字1</value>         <value>數字2</value>         <value>數字3</value>       </list>     </property>     <!-- list集合 -->     <property name="list">       <list>         <value>金在中</value>         <value>王杰</value>       </list>     </property>     <!-- set集合 -->     <property name="sets">       <set>         <value>哈哈</value>         <value>呵呵</value>       </set>     </property>     <!-- map集合 -->     <property name="map">       <map>         <entry key="aa" value="rainbow"/>         <entry key="bb" value="hellowvenus"/>       </map>     </property>     <!-- 屬性文件 -->     <property name="pro">       <props>         <prop key="username">root</prop>         <prop key="password">123</prop>       </props>     </property>   </bean>

3)測試

 /**   * 測試注入集合   */  @Test  public void run3(){    ApplicationContext ac=new ClassPathXmlApplicationContext("applicationContext.xml");    User user= (User) ac.getBean("user");    System.out.println(user);  }

5.怎么分模塊開發

在主配置文件加入<import>標簽(假如此時在com.clj.test包下定義了一個配置文件applicationContext2.xml)

<!-- 分模塊開發之引入其他配置文件 -->   <import resource="com/clj/test/applicationContext2.xml"/>

五、詳解Spring框架的IOC之注解方式

1、入門

1).導入jar包

除了先前6個包,玩注解還需一個spring-aop包

Spring,框架,Spring框架  

2).持久層和實現層(這里忽略接口)

持久層

package com.clj.demo1;import org.springframework.context.annotation.Scope;import org.springframework.stereotype.Component;import org.springframework.stereotype.Repository;/** * UserDaoImpl交給IOC的容器管理 * @author Administrator * */public class UserDaoImpl implements UserDao{  @Override  public void save() {    System.out.println("保存客戶。。");      }}

業務層

package com.clj.demo1;import javax.annotation.PostConstruct;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.beans.factory.annotation.Qualifier;import org.springframework.beans.factory.annotation.Value;import org.springframework.stereotype.Component;public class UserServiceImpl implements UserService{  @Override  public void sayHello() {    System.out.println("Hello spring");  }}

3).定義配置文件

此時約束條件需添加context約束,并添加組件掃描

<?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"> <!-- bean definitions here -->  <!-- 開啟注解掃面 :base-package指定掃面對 包-->  <context:component-scan base-package="com.clj.demo1"/></beans>

4)在實現類中添加注解

/** * 組件注解,可以用來標記當前的類 * 類似<bean id="userService" class="com.clj.demo1.UserServiceImpl"> * value表示給該類起個別名 */@Component(value="userService")public class UserServiceImpl implements UserService{      //省略}

5)編寫測試

  /**   * 注解方式   */  @Test  public void run2(){    ApplicationContext ac=new ClassPathXmlApplicationContext("applicationContext.xml");    UserService us=(UserService) ac.getBean("userService");    us.sayHello();  }

2.關于bean管理常用屬性

1. @Component:組件.(作用在類上)  最原始的注解,所有需要注解的類都寫這個沒問題,他是通用的

2. Spring中提供@Component的三個衍生注解:(功能目前來講是一致的)
    * @Controller       -- 作用在WEB層
    * @Service          -- 作用在業務層
    * @Repository       -- 作用在持久層
    * 說明:這三個注解是為了讓標注類本身的用途清晰,Spring在后續版本會對其增強

3. 屬性注入的注解(說明:使用注解注入的方式,可以不用提供set方法)
    * 如果是注入的普通類型,可以使用value注解
    * @Value             -- 用于注入普通類型
    * 如果注入的是對象類型,使用如下注解
        * @Autowired        -- 默認按類型進行自動裝配   匹配的是類型,與注入類的類名無關
            * 如果想按名稱注入
            * @Qualifier    -- 強制使用名稱注入            必須與Autowired一起用,指定類名,與注入的類名有關
        * @Resource         -- 相當于@Autowired和@Qualifier一起使用
        * 強調:Java提供的注解
        * 屬性使用name屬性

4. Bean的作用范圍注解

    * 注解為@Scope(value="prototype"),作用在類上。值如下:

        * singleton     -- 單例,默認值

        * prototype     -- 多例

5. Bean的生命周期的配置(了解)

    * 注解如下:

        * @PostConstruct    -- 相當于init-method

        * @PreDestroy       -- 相當于destroy-method

1.演示屬性對象注解

條件:采用掃描的方式將屬性(name)和對象(userDaoImpl)注入到業務層中

1)持久層開啟注解掃描Repository

//@Component(value="userDao")通用類注解@Repository(value="ud")public class UserDaoImpl implements UserDao{  @Override  public void save() {    System.out.println("保存客戶。。");    }}

2)業務層針對屬性和對象提供注解

package com.clj.demo1;import javax.annotation.PostConstruct;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.beans.factory.annotation.Qualifier;import org.springframework.beans.factory.annotation.Value;import org.springframework.stereotype.Component;/** * 組件注解,可以用來標記當前的類 * 類似<bean id="userService" class="com.clj.demo1.UserServiceImpl"> * value表示給該類起個別名 *///@Scope(value="grototype")多列的(singletype為單列)@Component(value="userService")public class UserServiceImpl implements UserService{  //屬性注解:相當于給name屬性注入指定的字符串,setName方法可以省略不寫  @Value(value="佳先森")  private String name;    /**   * 引用注入方式一:Autowired()   * 引用注入方式二:Autowired()+Qualifier   * 引用注入方式三:@Resource(name="userDao") java方式,按名稱識別注入   */  //Autowired()按類型自動裝配注入(缺點:因為是按類型匹配,所以不是很準確)  @Autowired()  @Qualifier(value="ud") //按名稱注入,得與Autowired一起用,兩者一起能指定類  private UserDao userDao;  //注意Qualifier中的value是指定UserDaoImpl類名頂上的注解名,也可以指定配置文件中bean的id名      /*public void setName(String name) {    this.name = name;  }*/  @Override  public void sayHello() {    System.out.println("Hello spring"+name);    userDao.save();  }  //@PostConstruct標簽用于action生命周期中初始化的注解  @PostConstruct  public void init(){    System.out.println("初始化...");  }}

3)配置文件只需要開啟全部掃描即可

<?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"> <!-- bean definitions here -->  <!-- 開啟注解掃面 :base-package指定掃面對 包-->  <context:component-scan base-package="com.clj.demo1"/></beans>

注意:至于集合還是推薦使用配置文件方式

2.Spring框架整合JUnit單元測試

1)添加單元測試所需依賴包spring-test.jar

Spring,框架,Spring框架  

注意:基于myeclipes自帶Junit環境,但是有時因為版本問題,可能需要比較新的Junit環境,這里我在網上下了一個教新的 Junit-4.9的jar包,如果myeclipes較新的話無須考慮

2)編寫測試類,添加相對應的注解

@RunWith與@ContextConfiguration(此是用于加載配置文件,因為默認從WebRoot路徑為一級目錄,加上此是認定src為一級目錄)

package com.clj.demo2;import javax.annotation.Resource;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.test.context.ContextConfiguration;import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;import com.clj.demo1.UserService;@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration("classpath:applicationContext.xml")public class Demo2 {  @Resource(name="userService")  private UserService userService;  @Test  public void run1(){    userService.sayHello();  }}

六.spring框架之AOP

1.什么是AOP

        * 在軟件業,AOP為Aspect Oriented Programming的縮寫,意為:面向切面編程,功能模塊化

        * AOP是一種編程范式,隸屬于軟工范疇,指導開發者如何組織程序結構

        * AOP最早由AOP聯盟的組織提出的,制定了一套規范.Spring將AOP思想引入到框架中,必須遵守AOP聯盟的規范

        * 通過預編譯方式和運行期動態代理實現程序功能的統一維護的一種技術

        * AOP是OOP的延續,是軟件開發中的一個熱點,也是Spring框架中的一個重要內容,是函數式編程的一種衍生范型

        * 利用AOP可以對業務邏輯的各個部分進行隔離,從而使得業務邏輯各部分之間的耦合度降低,提高程序的可重用性,同時提高了開發的效率

      AOP采取橫向抽取機制,取代了傳統縱向繼承體系重復性代碼(性能監視、事務管理、安全檢查、緩存)

2. 為什么要學習AOP

        * 可以在不修改源代碼的前提下,對程序進行增強?。。楣潭ǖ姆椒ㄉ梢粋€代理,在訪問該方法之前,先進入代理,在代理中,可以編寫更多的功能,使之方法的功能更強,使得程序進行增        強)

Aop:面向切面編程,將一切事模塊化,每個模塊比較獨立,模塊可以共用(相同的),不同的格外自定義。用此替代傳統的面向縱向編程,提高程序的可重用性

3.AOP的實現(實現原理)

Aop的實現包含兩種代理方式<1>實現類接口:采用JDK動態代理<2>未實現類接口:采用CGLIB動態代理

   1.實現JDK動態代理

      1)定義持久層接口實現類

package com.clj.demo3;public interface UserDao {  public void save();  public void update();}
package com.clj.demo3;public class UserDaoImpl implements UserDao {  @Override  public void save() {    System.out.println("保存用戶");  }  @Override  public void update() {    System.out.println("修改用戶");  }}

   2)定義JDK動態代理工具類

此工具類是在執行持久層save方法時增加一些功能,在開發中做到在不更改源碼情況下增強某方法

package com.clj.demo3;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;/** * 使用JDK的方式生成代理對象(演示AOP原理) * @author Administrator * */public class MyProxyUtils {  public static UserDao getProxy(final UserDao dao){    //使用Proxy類生成代理對象    UserDao proxy=(UserDao)Proxy.newProxyInstance(dao.getClass().getClassLoader() , dao.getClass().getInterfaces(),new InvocationHandler() {      //只要代理對象一執行,invoke方法就會執行一次      public Object invoke(Object proxy, Method method, Object[] args)          throws Throwable {        //proxy代表當前代理對象        //method當前對象執行的方法        //args封裝的參數        //讓到類的save或者update方法正常執行下去        if("save".equals(method.getName())){          System.out.println("執行了保存");          //開啟事務        }        return method.invoke(dao, args);      }    });    return proxy;  }}

 3)測試

package com.clj.demo3;import org.junit.Test;public class Demo1 {  @Test  public void run1(){    //獲取目標對象    UserDao dao=new UserDaoImpl();    dao.save();    dao.update();    System.out.println("===============");    //使用工具類,獲取到代理對象    UserDao proxy=MyProxyUtils.getProxy(dao);    //調用代理對象的方法    proxy.save();    proxy.update();      }}

 2.實現CGLIB技術

 1)定義持久層,此時沒有接口

package com.clj.demo4;public class BookDaoImpl {  public void save(){    System.out.println("保存圖書");  }  public void update(){    System.out.println("修改圖書");  }}

  2)編寫工具類

package com.clj.demo4;import java.lang.reflect.Method;import org.springframework.cglib.proxy.Enhancer;import org.springframework.cglib.proxy.MethodInterceptor;import org.springframework.cglib.proxy.MethodProxy;/** * Cglib代理方式實現原理 * @author Administrator * */public class MyCglibUtils {  /**   * 使用CGLIB方式生成代理對象   * @return   */  public static BookDaoImpl getProxy(){    Enhancer enhancer=new Enhancer();    //設置父類    enhancer.setSuperclass(BookDaoImpl.class);    //設置回調函數    enhancer.setCallback(new MethodInterceptor() {            @Override      public Object intercept(Object obj, Method method, Object[] objs,          MethodProxy methodProxy) throws Throwable {        if(method.getName().equals("save")){        System.out.println("我保存了");        System.out.println("代理對象執行了");    }        return methodProxy.invokeSuper(obj, objs);//是方法執行下去      }    });    //生成代理對象    BookDaoImpl proxy=(BookDaoImpl) enhancer.create();    return proxy;  }}

  3)編寫測試類

package com.clj.demo4;import org.junit.Test;public class Demo1 {  @Test  public void run1(){    //目標對象    BookDaoImpl dao=new BookDaoImpl();    dao.save();    dao.update();    System.out.println("==========");    BookDaoImpl proxy=MyCglibUtils.getProxy();    proxy.save();    proxy.update();  }}

 3、Spring基于AspectJ的AOP的開發(配置文件方式)

Spring,框架,Spring框架  

  1)部署環境,導入相對應的jar包

Spring,框架,Spring框架  

 2)創建配置文件,并引入AOP約束

<beans xmlns="http://www.springframework.org/schema/beans"      xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"      xmlns:aop="http://www.springframework.org/schema/aop"      xsi:schemaLocation="      http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd      http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">

 3)創建接口和實現類

package com.clj.demo5;public interface CustomerDao {  public void save();  public void update();}
package com.clj.demo5;/** * 采用配置文件的方式 詮釋AOP * @author Administrator * */public class CustomerDaoImpl implements CustomerDao {  @Override  public void save() {    //模擬異常    //int a=10/0;    System.out.println("保存客戶了啊");  }  @Override  public void update() {    // TODO Auto-generated method stub    System.out.println("更新客戶了啊");  }}

 4)定義切面類

package com.clj.demo5;import org.aspectj.lang.ProceedingJoinPoint;/** * 切面類:切入點+通知 * @author Administrator * */public class MyAspectXml {  /**   * 通知(具體的增強)   */  public void log(){    System.out.println("記錄日志");  }  /**   * 方法執行成功或者異常都會執行   */  public void after(){    System.out.println("最終通知");  }  /**   * 方法執行之后,執行后置通知,如果程序出現異常,后置通知不會執行   */  public void afterReturn(){    System.out.println("后置通知");  }  /**   * 方法執行之后,如果程序有異常,才會執行異常通知   */  public void afterThrowing(){    System.out.println("異常通知");  }  /**   * 環繞通知:方法執行之前和方法執行之后進行通知,   * 默認情況下,目標對象的方法不能執行的,需要手動讓目標對象執行   */  public void around(ProceedingJoinPoint joinPoint){    System.out.println("環繞通知1");    //手動讓目標對象的方法執行    try {      joinPoint.proceed();    } catch (Throwable e) {      // TODO Auto-generated catch block      e.printStackTrace();    }    System.out.println("環繞通知2");  }}

 5)注入實現類和切面類

<?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:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd    http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <!-- bean definitions here -->  <!-- 配置客戶的dao -->  <bean id="customerDao" class="com.clj.demo5.CustomerDaoImpl"/>  <!-- 編寫切面類配置好 -->  <bean id="myAspectXml" class="com.clj.demo5.MyAspectXml"/>  <!-- 配置AOP -->  <aop:config>    <!-- 配置切面類:切入點+通知 (類型)-->    <aop:aspect ref="myAspectXml">      <!-- 配置前置通知,save方法執行之前,增強方法會執行 -->      <!-- 切入點表達式:execution(public void com.clj.demo5.CustomerDaoImpl.save()) -->      <!-- 切入點表達式:        1.execution()固定的,必寫        2.public可以省略不寫        3.返回值   必寫,嚴格根據切入點方法而定,否則增強方法不會執行,可以用*代替,表示任意的返回值        4.包名   必寫,可以用*代替(如:*..*(默認所有包); com.clj.*)        5.類名   必寫,可以部分用*(如*DaoImpl表示以'DaoImpl'結尾的持久層實現類),但不建議用*代替整個類名        6.方法   必寫,可以部分用*(如save*表示以'save'開頭的方法),但不建議用*代替整個類名        7.方法參數 根據實際方法而定,可以用'..'表示有0或者多個參數       -->      <!-- <aop:before method="log" pointcut="execution(public void com.clj.*.CustomerDaoImpl.save(..))"/> -->      <aop:before method="log" pointcut="execution(* *..*.*DaoImpl.save*(..))"/>    </aop:aspect>  </aop:config></beans>

 6)測試

 package com.clj.demo5;import javax.annotation.Resource;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.test.context.ContextConfiguration;import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration("classpath:applicationContext.xml")public class Demo1 {  @Resource(name="customerDao")  private CustomerDao customerDao;  @Test  public void run(){    customerDao.save();    customerDao.update();  }}

擴展:切面類升級

<?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:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="    http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd    http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <!-- bean definitions here -->  <bean id="myAspectXml" class="com.clj.demo5.MyAspectXml"/>  <aop:config>    <aop:aspect ref="myAspectXml">      <!-- 配置最終通知       <aop:after method="after" pointcut="execution(* *..*.*DaoImpl.save*(..))"/>-->      <!-- 配置后置通知       <aop:after-returning method="afterReturn" pointcut="execution(* *..*.*DaoImpl.save*(..))"/>-->      <!-- 配置異常通知       <aop:after-throwing method="afterThrowing" pointcut="execution(* *..*.*DaoImpl.save*(..))"/>-->      <aop:around method="around" pointcut="execution(* *..*.*DaoImpl.update*(..))"/>    </aop:aspect>  </aop:config></beans>

4、Spring框架AOP之注解方式

 1)創建接口和實現類

package com.clj.demo1;public interface CustomerDao {  public void save();  public void update();}
package com.clj.demo1;public class CustomerDaoImpl implements CustomerDao{  @Override  public void save() {    // TODO Auto-generated method stub    System.out.println("保存客戶..");  }  @Override  public void update() {    // TODO Auto-generated method stub    System.out.println("更新客戶");  }}

2)定義切面類

package com.clj.demo1;import org.aspectj.lang.ProceedingJoinPoint;import org.aspectj.lang.annotation.After;import org.aspectj.lang.annotation.Around;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Before;import org.aspectj.lang.annotation.Pointcut;/** * 注解方式的切面類 * @Aspect表示定義為切面類 */@Aspectpublic class MyAspectAnno {  //通知類型:@Before前置通知(切入點的表達式)  @Before(value="execution(public *   com.clj.demo1.CustomerDaoImpl.save())")  public void log(){    System.out.println("記錄日志。。");  }  //引入切入點  @After(value="MyAspectAnno.fun()")  public void after(){    System.out.println("執行之后");  }  @Around(value="MyAspectAnno.fun()")  public void around(ProceedingJoinPoint joinPoint){    System.out.println("環繞通知1");    try {      //讓目標對象執行      joinPoint.proceed();    } catch (Throwable e) {      // TODO Auto-generated catch block      e.printStackTrace();    }    System.out.println("環繞通知2");  }  //自定義切入點  @Pointcut(value="execution(public *   com.clj.demo1.CustomerDaoImpl.save())")  public void fun(){      }}

3)配置切面類和實現類,并開啟自動代理

<?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:tx="http://www.springframework.org/schema/tx"  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  http://www.springframework.org/schema/aop  http://www.springframework.org/schema/aop/spring-aop.xsd  http://www.springframework.org/schema/tx   http://www.springframework.org/schema/tx/spring-tx.xsd">  <!-- 開啟自動注解代理-->  <aop:aspectj-autoproxy/>   <!-- 配置目標對象 -->  <bean id="customerDao" class="com.clj.demo1.CustomerDaoImpl"/>  <!-- 配置切面類 -->  <bean id="myAspectAnno" class="com.clj.demo1.MyAspectAnno"/></beans>

七、Spring之JDBC

spring提供了JDBC模板:JdbcTemplate類

1.快速搭建

 1)部署環境

這里在原有的jar包基礎上,還要添加關乎jdbc的jar包,這里使用的是mysql驅動

Spring,框架,Spring框架  

 2)配置內置連接池,將連接數據庫程序交給框架管理,并配置Jdbc模板類

<?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:tx="http://www.springframework.org/schema/tx"  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  http://www.springframework.org/schema/aop  http://www.springframework.org/schema/aop/spring-aop.xsd  http://www.springframework.org/schema/tx   http://www.springframework.org/schema/tx/spring-tx.xsd">  <!-- 先配置連接池(內置) -->  <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">    <property name="driverClassName" value="com.mysql.jdbc.Driver"/>    <property name="url" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="username" value="root"/>    <property name="password" value="root"/>  </bean>  <!-- 配置JDBC的模板類-->  <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">    <property name="dataSource" ref="dataSource"/>  </bean></beans>

 3)測試

package com.clj.demo2;import java.sql.ResultSet;import java.sql.SQLException;import java.util.List;import javax.annotation.Resource;import org.apache.commons.dbcp.BasicDataSource;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.cglib.beans.BeanMap;import org.springframework.jdbc.core.JdbcTemplate;import org.springframework.jdbc.core.RowMapper;import org.springframework.test.context.ContextConfiguration;import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;/** * 測試JDBC的模板類,使用IOC的方式 * @author Administrator * */@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration("classpath:applicationContext.xml")public class Demo2 {  @Resource(name="jdbcTemplate")  private JdbcTemplate jdbcTemplate;  /**   * 插入   */  @Test  public void run1(){    String sql="insert into t_account values(null,?,?)";    jdbcTemplate.update(sql,"李釔林",10000);  }  /**   * 更新   */  @Test  public void run2(){    String sql="update t_account set name=? where id=?";    jdbcTemplate.update(sql,"李釔林",1);  }  /**   * 刪除   */  @Test  public void run3(){    String sql="delete from t_account where id=?";    jdbcTemplate.update(sql,4);  }  /**   * 測試查詢,通過主鍵來查詢一條記錄   */  @Test  public void run4(){    String sql="select * from t_account where id=?";    Account ac=jdbcTemplate.queryForObject(sql, new BeanMapper(),1);    System.out.println(ac);  }  /**   * 查詢所有   */  @Test  public void run5(){    String sql="select * from t_account";    List<Account> ac=jdbcTemplate.query(sql,new BeanMapper());    System.out.println(ac);  }}/** * 定義內部類(手動封裝數據(一行一行封裝數據,用于查詢所有) * @author Administrator * */class BeanMapper implements RowMapper<Account>{  @Override  public Account mapRow(ResultSet rs, int rowNum) throws SQLException {    Account ac=new Account();    ac.setId(rs.getInt("id"));    ac.setName(rs.getString("name"));    ac.setMoney(rs.getDouble("money"));    return ac;  }  }

2、配置開源連接池

一般現在企業都是用一些主流的連接池,如c3p0和dbcp

首先配置dbcp

1)導入dbcp依賴jar包

Spring,框架,Spring框架  

2)編寫配置文件

<!-- 配置DBCP開源連接池-->   <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">    <property name="driverClassName" value="com.mysql.jdbc.Driver"/>    <property name="url" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="username" value="root"/>    <property name="password" value="root"/>  </bean>

將模板類中引入的內置類datasource改為開源連接池的

3)編寫測試類

配置c3p0

 1)導入c3p0依賴jar包

Spring,框架,Spring框架  

2)配置c3p0

<!-- 配置C3P0開源連接池 -->  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="com.mysql.jdbc.Driver"/>    <property name="jdbcUrl" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="user" value="root"/>    <property name="password" value="root"/>  </bean>

將模板類中引入的內置類datasource改為開源連接池的

3)編寫測試類

八、Spring之事務

1、什么是事務

數據庫事務(Database Transaction) ,是指作為單個邏輯工作單元執行的一系列操作,要么完全地執行,要么完全地不執行。 事務處理可以確保除非事務性單元內的所有操作都成功完成,否則不會永久更新面向數據的資源。通過將一組相關操作組合為一個要么全部成功要么全部失敗的單元,可以簡化錯誤恢復并使應用程序更加可靠。一個邏輯工作單元要成為事務,必須滿足所謂的ACID(原子性、一致性、隔離性和持久性)屬性。事務是數據庫運行中的邏輯工作單位,由DBMS中的事務管理子系統負責事務的處理。

2、怎么解決事務安全性問題

讀問題解決,設置數據庫隔離級別;寫問題解決可以使用 悲觀鎖和樂觀鎖的方式解決

3、快速開發

方式一:調用模板類,將模板注入持久層

1)編寫相對應的持久層和也外層,這里省略接口   

package com.clj.demo3;import org.springframework.jdbc.core.JdbcTemplate;import org.springframework.jdbc.core.support.JdbcDaoSupport;public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao{  // 方式一:將jdbc模板類注入到配置文件中,直接在持久層寫模板類   private JdbcTemplate jdbcTemplate;   public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {     this.jdbcTemplate = jdbcTemplate;   }    public void outMoney(String out, double money) {    String sql="update t_account set money=money-? where name=?";    jdbcTemplate().update(sql,money,out);  }    public void inMoney(String in, double money) {    String sql="update t_account set money=money+? where name=?";    jdbcTemplate().update(sql,money,in);  }  }
package com.clj.demo4;import org.springframework.transaction.TransactionStatus;import org.springframework.transaction.support.TransactionCallbackWithoutResult;import org.springframework.transaction.support.TransactionTemplate;public class AccountServiceImpl implements AccountService{  //采用的是配置文件注入方式,必須提供set方法  private AccountDao accountDao;  public void setAccountDao(AccountDao accountDao) {    this.accountDao = accountDao;  }  @Override  public void pay(String out, String in, double money) {    // TODO Auto-generated method stub    accountDao.outMoney(out, money);    int a=10/0;    accountDao.inMoney(in, money);  }}

2)配置相對應的配置文件

<?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:tx="http://www.springframework.org/schema/tx"  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  http://www.springframework.org/schema/aop  http://www.springframework.org/schema/aop/spring-aop.xsd  http://www.springframework.org/schema/tx   http://www.springframework.org/schema/tx/spring-tx.xsd"><!-- 配置C3P0開源連接池 -->  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="com.mysql.jdbc.Driver"/>    <property name="jdbcUrl" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="user" value="root"/>    <property name="password" value="root"/>  </bean><!-- 配置JDBC的模板類 -->  <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">    <property name="dataSource" ref="dataSource"/>  </bean><!-- 配置業務層和持久層 -->  <bean id="accountService" class="com.clj.demo3.AccountServiceImpl">    <property name="accountDao" ref="accountDao"/>  </bean><bean id="accountDao" class="com.clj.demo3.AccountDaoImpl">    <!-- 注入模板類-->    <property name="jdbcTemplate" ref="jdbcTemplate"/>       <property name="dataSource" ref="dataSource"/>  </bean></beans>

3)測試類

package com.clj.demo3;import javax.annotation.Resource;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.test.context.ContextConfiguration;import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration("classpath:applicationContext.xml")public class Demo1 {  @Resource(name="accountService")  private AccountService accountService;  @Test  public void Demo1(){    //調用支付的方法    accountService.pay("佳先森","李釔林",100);  }}

方式二:持久層繼承JdbcDaoSupport接口,此接口封裝了模板類jdbcTemplate

Spring,框架,Spring框架

1)編寫配置文件

<?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:tx="http://www.springframework.org/schema/tx"  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  http://www.springframework.org/schema/aop  http://www.springframework.org/schema/aop/spring-aop.xsd  http://www.springframework.org/schema/tx   http://www.springframework.org/schema/tx/spring-tx.xsd">  <!-- 配置C3P0開源連接池 -->  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="com.mysql.jdbc.Driver"/>    <property name="jdbcUrl" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="user" value="root"/>    <property name="password" value="root"/>  </bean>  <!-- 配置業務層和持久層 -->  <bean id="accountService" class="com.clj.demo3.AccountServiceImpl">    <property name="accountDao" ref="accountDao"/>  </bean>  <bean id="accountDao" class="com.clj.demo3.AccountDaoImpl">      <property name="dataSource" ref="dataSource"/>  </bean></beans>

2)更改持久層

package com.clj.demo3;import org.springframework.jdbc.core.JdbcTemplate;import org.springframework.jdbc.core.support.JdbcDaoSupport;public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao{  //方式一:將jdbc模板類注入到配置文件中,直接在持久層寫模板類//  private JdbcTemplate jdbcTemplate;//  public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {//    this.jdbcTemplate = jdbcTemplate;//  }  //方式二:持久層繼承JdbcDaoSupport,它里面封轉了模板類,配置文件持久層無需注入模板類,也不需要配置模板類  public void outMoney(String out, double money) {    //jdbcTemplate.update(psc);    String sql="update t_account set money=money-? where name=?";    this.getJdbcTemplate().update(sql,money,out);  }    public void inMoney(String in, double money) {    String sql="update t_account set money=money+? where name=?";    this.getJdbcTemplate().update(sql,money,in);  }  }

3)更改業務層

package com.clj.demo4;import org.springframework.transaction.TransactionStatus;import org.springframework.transaction.support.TransactionCallbackWithoutResult;import org.springframework.transaction.support.TransactionTemplate;public class AccountServiceImpl implements AccountService{  //采用的是配置文件注入方式,必須提供set方法  private AccountDao accountDao;  public void setAccountDao(AccountDao accountDao) {    this.accountDao = accountDao;  }  @Override  public void pay(String out, String in, double money) {    // TODO Auto-generated method stub    accountDao.outMoney(out, money);    int a=10/0;    accountDao.inMoney(in, money);  }  }

4)測試類和上述一樣

4、spring事務管理

Spring為了簡化事務管理的代碼:提供了模板類 TransactionTemplate,手動編程的方式來管理事務,只需要使用該模板類即可??!

九、Spring框架的事務管理之編程式的事務管理

1、手動編程方式事務(了解原理)

1)快速部署,搭建配置文件,配置事務管理和事務管理模板,并在持久層注入事務管理模板

配置事務管理器

  <!-- 配置平臺事務管理器 -->  <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">    <property name="dataSource" ref="dataSource"/>  </bean>

配置事務管理模板

<bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">    <property name="transactionManager" ref="transactionManager"/></bean>

將管理模板注入業務層

<bean id="accountService" class="com.clj.demo3.AccountServiceImpl">    <property name="accountDao" ref="accountDao"/>    <property name="transactionTemplate" ref="transactionTemplate"/></bean>

全部代碼:

<?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:tx="http://www.springframework.org/schema/tx"  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  http://www.springframework.org/schema/aop  http://www.springframework.org/schema/aop/spring-aop.xsd  http://www.springframework.org/schema/tx   http://www.springframework.org/schema/tx/spring-tx.xsd">    <!-- 配置C3P0開源連接池 -->  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="com.mysql.jdbc.Driver"/>    <property name="jdbcUrl" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="user" value="root"/>    <property name="password" value="root"/>  </bean>   <!-- 配置業務層和持久層 -->  <bean id="accountService" class="com.clj.demo3.AccountServiceImpl">    <property name="accountDao" ref="accountDao"/>    <property name="transactionTemplate" ref="transactionTemplate"/>  </bean>  <bean id="accountDao" class="com.clj.demo3.AccountDaoImpl">      <property name="dataSource" ref="dataSource"/>  </bean>  <!-- 配置平臺事務管理器 -->  <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">    <property name="dataSource" ref="dataSource"/>  </bean>  <!-- 手動編碼方式,提供了模板類,使用該類管理事務比較簡單-->  <bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">    <property name="transactionManager" ref="transactionManager"/>  </bean></beans>

2)在業務層使用模板事務管理

package com.clj.demo3;import org.springframework.transaction.TransactionStatus;import org.springframework.transaction.support.TransactionCallbackWithoutResult;import org.springframework.transaction.support.TransactionTemplate;public class AccountServiceImpl implements AccountService{  //采用的是配置文件注入方式,必須提供set方法  private AccountDao accountDao;  //注入事務模板類  private TransactionTemplate transactionTemplate;  public void setAccountDao(AccountDao accountDao) {    this.accountDao = accountDao;  }  public void setTransactionTemplate(TransactionTemplate transactionTemplate) {    this.transactionTemplate = transactionTemplate;  }  /**   * 轉賬的方法   */  public void pay(final String out,final String in, final double money) {    transactionTemplate.execute(new TransactionCallbackWithoutResult() {      //事務的執行,如果沒有問題,提交,如果楚翔異常,回滾      protected void doInTransactionWithoutResult(TransactionStatus arg0) {        // TODO Auto-generated method stub        accountDao.outMoney(out, money);        int a=10/0;        accountDao.inMoney(in, money);      }    });  }}

3)測試類和上一致

package com.clj.demo4;import javax.annotation.Resource;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.test.context.ContextConfiguration;import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration("classpath:applicationContext2.xml")public class Demo2 {  @Resource(name="accountService")  private AccountService accountService;  @Test  public void Demo1(){    //調用支付的方法    accountService.pay("佳先森","李釔林",100);  }}

十、Spring框架的事務管理之聲明式事務管理,即通過配置文件來完成事務管理(AOP思想)

申明式事務有兩種方式:基于AspectJ的XML方式;基于AspectJ的注解方式

1、XML方式

1)配置配置文件

需要配置平臺事務管理

<!-- 配置C3P0開源連接池 -->  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="com.mysql.jdbc.Driver"/>    <property name="jdbcUrl" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="user" value="root"/>    <property name="password" value="root"/>  </bean>  <!-- 配置平臺事務管理器 -->  <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">    <property name="dataSource" ref="dataSource"/>  </bean>

配置事務增強

<tx:advice id="myAdvice" transaction-manager="transactionManager">    <tx:attributes>      <!-- 給方法設置數據庫屬性(隔離級別,傳播行為) -->      <!--propagation事務隔離級別:一般采用默認形式:tx:method可以設置多個 -->      <tx:method name="pay" propagation="REQUIRED"/>    </tx:attributes>  </tx:advice>

aop切面類

<aop:config>    <!-- aop:advisor,是spring框架提供的通知-->    <aop:advisor advice-ref="myAdvice" pointcut="execution(public * com.clj.demo4.AccountServiceImpl.pay(..))"/>  </aop:config>

全部代碼

<?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:tx="http://www.springframework.org/schema/tx"  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  http://www.springframework.org/schema/aop  http://www.springframework.org/schema/aop/spring-aop.xsd  http://www.springframework.org/schema/tx   http://www.springframework.org/schema/tx/spring-tx.xsd">  <!-- 配置C3P0開源連接池 -->  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="com.mysql.jdbc.Driver"/>    <property name="jdbcUrl" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="user" value="root"/>    <property name="password" value="root"/>  </bean>  <!-- 配置平臺事務管理器 -->  <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">    <property name="dataSource" ref="dataSource"/>  </bean>  <!-- 申明式事務(采用XML文件的方式) -->  <!-- 先配置通知 -->  <tx:advice id="myAdvice" transaction-manager="transactionManager">    <tx:attributes>      <!-- 給方法設置數據庫屬性(隔離級別,傳播行為) -->      <!--propagation事務隔離級別:一般采用默認形式:tx:method可以設置多個 -->      <tx:method name="pay" propagation="REQUIRED"/>    </tx:attributes>  </tx:advice>  <!-- 配置AOP:如果是自己編寫的AOP,使用aop:aspect配置,使用的是Spring框架提供的通知 -->  <aop:config>    <!-- aop:advisor,是spring框架提供的通知-->    <aop:advisor advice-ref="myAdvice" pointcut="execution(public * com.clj.demo4.AccountServiceImpl.pay(..))"/>  </aop:config>    <!-- 配置業務層和持久層 -->  <bean id="accountService" class="com.clj.demo4.AccountServiceImpl">    <property name="accountDao" ref="accountDao"/>  </bean>  <bean id="accountDao" class="com.clj.demo4.AccountDaoImpl">    <property name="dataSource" ref="dataSource"/>  </bean></beans>

2)編寫持久層和業務層(省略接口)

package com.clj.demo5;import org.springframework.jdbc.core.JdbcTemplate;import org.springframework.jdbc.core.support.JdbcDaoSupport;public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao{  //方式一:將jdbc模板類注入到配置文件中,直接在持久層寫模板類//  private JdbcTemplate jdbcTemplate;//  public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {//    this.jdbcTemplate = jdbcTemplate;//  }  //方式二:持久層繼承JdbcDaoSupport,它里面封轉了模板類,配置文件持久層無需注入模板類,也不需要配置模板類  public void outMoney(String out, double money) {    //jdbcTemplate.update(psc);    String sql="update t_account set money=money-? where name=?";    this.getJdbcTemplate().update(sql,money,out);  }    public void inMoney(String in, double money) {    String sql="update t_account set money=money+? where name=?";    this.getJdbcTemplate().update(sql,money,in);  }  }
package com.clj.demo5;import org.springframework.transaction.TransactionStatus;import org.springframework.transaction.annotation.Transactional;import org.springframework.transaction.support.TransactionCallbackWithoutResult;import org.springframework.transaction.support.TransactionTemplate;public class AccountServiceImpl implements AccountService{  //采用的是配置文件注入方式,必須提供set方法  private AccountDao accountDao;  public void setAccountDao(AccountDao accountDao) {    this.accountDao = accountDao;  }  @Override  public void pay(String out, String in, double money) {    // TODO Auto-generated method stub    accountDao.outMoney(out, money);    int a=10/0;    accountDao.inMoney(in, money);  }  }

3)測試類

package com.clj.demo4;import javax.annotation.Resource;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.test.context.ContextConfiguration;import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration("classpath:applicationContext2.xml")public class Demo2 {  @Resource(name="accountService")  private AccountService accountService;  @Test  public void Demo1(){    //調用支付的方法    accountService.pay("佳先森","李釔林",100);  }}

2、注解方式

1)配置配置文件

配置事務管理

<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="com.mysql.jdbc.Driver"/>    <property name="jdbcUrl" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="user" value="root"/>    <property name="password" value="root"/>  </bean>  <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">    <property name="dataSource" ref="dataSource"/>  </bean>

開啟注釋事務

  <!-- 開啟事務的注解 -->  <tx:annotation-driven transaction-manager="transactionManager"/>

全部代碼

<?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:tx="http://www.springframework.org/schema/tx"  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  http://www.springframework.org/schema/aop  http://www.springframework.org/schema/aop/spring-aop.xsd  http://www.springframework.org/schema/tx   http://www.springframework.org/schema/tx/spring-tx.xsd">  <!-- 配置C3P0開源連接池 -->  <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">    <property name="driverClass" value="com.mysql.jdbc.Driver"/>    <property name="jdbcUrl" value="jdbc:mysql://192.168.174.130:3306/SSH"/>    <property name="user" value="root"/>    <property name="password" value="root"/>  </bean>  <!-- 配置平臺事務管理器 -->  <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">    <property name="dataSource" ref="dataSource"/>  </bean>  <!-- 開啟事務的注解 -->  <tx:annotation-driven transaction-manager="transactionManager"/>    <!-- 配置業務層和持久層 -->  <bean id="accountService" class="com.clj.demo5.AccountServiceImpl">    <property name="accountDao" ref="accountDao"/>  </bean>  <bean id="accountDao" class="com.clj.demo5.AccountDaoImpl">    <property name="dataSource" ref="dataSource"/>  </bean></beans>

2)業務層增加@Transactional

package com.clj.demo5;import org.springframework.transaction.TransactionStatus;import org.springframework.transaction.annotation.Transactional;import org.springframework.transaction.support.TransactionCallbackWithoutResult;import org.springframework.transaction.support.TransactionTemplate;//在當前類加此注解表示當前類所有的全部都有事務@Transactionalpublic class AccountServiceImpl implements AccountService{  //采用的是配置文件注入方式,必須提供set方法  private AccountDao accountDao;  public void setAccountDao(AccountDao accountDao) {    this.accountDao = accountDao;  }  @Override  public void pay(String out, String in, double money) {    // TODO Auto-generated method stub    accountDao.outMoney(out, money);    int a=10/0;    accountDao.inMoney(in, money);  }  }

3)持久層不變

package com.clj.demo5;import org.springframework.jdbc.core.JdbcTemplate;import org.springframework.jdbc.core.support.JdbcDaoSupport;public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao{  //方式一:將jdbc模板類注入到配置文件中,直接在持久層寫模板類//  private JdbcTemplate jdbcTemplate;//  public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {//    this.jdbcTemplate = jdbcTemplate;//  }  //方式二:持久層繼承JdbcDaoSupport,它里面封轉了模板類,配置文件持久層無需注入模板類,也不需要配置模板類  public void outMoney(String out, double money) {    //jdbcTemplate.update(psc);    String sql="update t_account set money=money-? where name=?";    this.getJdbcTemplate().update(sql,money,out);  }    public void inMoney(String in, double money) {    String sql="update t_account set money=money+? where name=?";    this.getJdbcTemplate().update(sql,money,in);  }  }

4)測試類

package com.clj.demo5;import javax.annotation.Resource;import org.junit.Test;import org.junit.runner.RunWith;import org.springframework.test.context.ContextConfiguration;import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration("classpath:applicationContext3.xml")public class Demo3 { @Resource(name="accountService") private AccountService accountService; @Test public void Demo1(){  //調用支付的方法  accountService.pay("佳先森","李釔林",100); }}

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩欧美在线一区| 91国内产香蕉| 97超级碰在线看视频免费在线看| 国产精品精品久久久久久| 日韩精品视频在线免费观看| 成人精品aaaa网站| 国产成人在线播放| 亚洲视频在线免费观看| 久久精品国产久精国产思思| 欧美日韩在线一区| 国产精品igao视频| 九九精品在线视频| 国产日本欧美一区二区三区在线| 欧美日韩福利在线观看| 亚洲精品综合久久中文字幕| 国产成人精品av在线| 欧美老少做受xxxx高潮| 国产亚洲成精品久久| 亚洲女人初尝黑人巨大| 国产精品自拍偷拍视频| 一本大道久久加勒比香蕉| 亚洲第一二三四五区| 国产精品入口免费视频一| 欧美丰满少妇xxxxx做受| 国产精品美乳一区二区免费| 97精品国产97久久久久久春色| 久久精品国产视频| 国产一区二区视频在线观看| 亚洲自拍偷拍色图| 欧美电影电视剧在线观看| 亚洲一区中文字幕在线观看| 日本高清不卡在线| 成人黄色影片在线| 久久九九国产精品怡红院| 欧美日韩国产黄| 欧美日韩国产一区二区| 欧美怡春院一区二区三区| 国产999精品久久久影片官网| 日韩欧美在线免费| 日韩有码在线电影| 亚洲欧洲一区二区三区久久| 精品激情国产视频| 欧美综合第一页| 亚洲大胆美女视频| 国产中文字幕日韩| 精品一区电影国产| 国产精品视频中文字幕91| 欧美刺激性大交免费视频| 日韩电影在线观看永久视频免费网站| 亚洲欧美色婷婷| 久久69精品久久久久久国产越南| 亚洲一区二区黄| 欧美日韩午夜视频在线观看| 国产精品久久久久久久久久三级| 激情久久av一区av二区av三区| 国产成人a亚洲精品| 深夜福利亚洲导航| 亚洲一区二区久久久久久久| 欧美整片在线观看| 欧美精品videosex牲欧美| 久久精品视频播放| 国产高清视频一区三区| 精品国产视频在线| 国产精品色悠悠| 久久影视电视剧免费网站清宫辞电视| 欧美在线视频免费观看| 97视频免费观看| 中文字幕不卡在线视频极品| 国产精品黄色影片导航在线观看| 精品偷拍一区二区三区在线看| 日韩在线精品视频| 欧美综合在线第二页| 亚洲字幕在线观看| 91经典在线视频| 岛国精品视频在线播放| 精品久久久久久久久久| 91av在线网站| 日韩美女免费观看| 日本欧美一二三区| 欧美日韩国产精品一区二区三区四区| 亚洲国产欧美一区二区丝袜黑人| 国产精品三级美女白浆呻吟| 欧美人在线视频| 久久天天躁狠狠躁夜夜躁| 色婷婷综合成人| 97香蕉久久超级碰碰高清版| 亚洲天堂一区二区三区| 久久久精品影院| 中文字幕在线看视频国产欧美在线看完整| 欧洲午夜精品久久久| 日本高清视频一区| 日韩经典中文字幕| 国产精品自产拍在线观| 亚洲成色www8888| 亚洲最大的网站| 中文.日本.精品| 久久久这里只有精品视频| 亚洲精品aⅴ中文字幕乱码| 国产精品人人做人人爽| xxxx欧美18另类的高清| 中文字幕欧美视频在线| 亚洲欧美日韩精品久久| 欧美在线日韩在线| 91在线国产电影| 色午夜这里只有精品| 国产精品青草久久久久福利99| 亚洲视频视频在线| 欧美成人三级视频网站| 欧美伊久线香蕉线新在线| 国产精品网站视频| 国产成人极品视频| 欧美成人精品在线视频| 欧美成年人视频网站欧美| 久久色精品视频| 亚洲电影免费观看高清完整版| 欧美与欧洲交xxxx免费观看| 在线国产精品视频| 欧美日韩精品中文字幕| 人人做人人澡人人爽欧美| 91欧美激情另类亚洲| 久久精品91久久香蕉加勒比| 日韩欧美中文字幕在线播放| 自拍偷拍亚洲区| 亚洲人永久免费| 欧美成人在线免费| 中文字幕亚洲天堂| 青青草原成人在线视频| 久久人体大胆视频| 中文字幕视频在线免费欧美日韩综合在线看| 精品视频久久久久久| 91精品久久久久久综合乱菊| 性夜试看影院91社区| 欧美电影免费播放| 91在线免费网站| 国产精品久久久av| 91在线中文字幕| 97欧美精品一区二区三区| 国产精品视频白浆免费视频| 色777狠狠综合秋免鲁丝| 亚洲第一精品久久忘忧草社区| 国产成人涩涩涩视频在线观看| 国产精品日韩av| 777国产偷窥盗摄精品视频| 亚洲高清不卡av| 狠狠色狠狠色综合日日小说| 日韩av最新在线观看| 性色av一区二区三区在线观看| 久久久999国产| 91精品国产九九九久久久亚洲| 色综合久久精品亚洲国产| 欧美wwwwww| 欧美xxxx14xxxxx性爽| 国产精品亚发布| 国产精品网红福利| 全球成人中文在线| 久久精品免费电影| 欧美自拍视频在线| 久久视频这里只有精品| 91亚洲精品久久久久久久久久久久| 欧美性猛交xxxx黑人| 欧美高清一级大片| 日韩精品小视频| 亚洲欧美国产另类| 茄子视频成人在线|