<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    隨筆 - 4  文章 - 10  trackbacks - 0
    <2025年5月>
    27282930123
    45678910
    11121314151617
    18192021222324
    25262728293031
    1234567

    常用鏈接

    留言簿(1)

    隨筆檔案

    文章分類

    文章檔案

    搜索

    •  

    最新評論

    閱讀排行榜

    評論排行榜

    首先我們來看看 Spring 參考文檔的 11.2.6. 執(zhí)行SQL語句 這里有個代碼片斷:

    import javax.sql.DataSource;
    import org.springframework.jdbc.core.JdbcTemplate;

    public class ExecuteAStatement {

        private JdbcTemplate jt;
        private DataSource dataSource;

        public void doExecute() {
            jt = new JdbcTemplate(dataSource);
            jt.execute("create table mytable (id integer, name varchar(100))");
        }

        public void setDataSource(DataSource dataSource) {
            this.dataSource = dataSource;
        }
    }

    這個就是普通的 Java 類, 再參考 11.2.4. DataSource接口, 這里的另一個代碼片斷:
    DriverManagerDataSource dataSource = new DriverManagerDataSource();
    dataSource.setDriverClassName("org.hsqldb.jdbcDriver");
    dataSource.setUrl("jdbc:hsqldb:hsql://localhost:");
    dataSource.setUsername("sa");
    dataSource.setPassword("");

    當然上面的連接方式可以配置成我們課程里面介紹的 MyEclipse Derby 的數(shù)據(jù)庫連接:
    org.apache.derby.jdbc.ClientDriver
    jdbc:derby://localhost:1527/myeclipse;create=true
    app
    app

    我們可以寫一個測試類來執(zhí)行代碼:

    import org.springframework.jdbc.datasource.DriverManagerDataSource;

    public class TestTemplate {
      public static void main(String[] args) {
        // 新建一個數(shù)據(jù)源對象
        DriverManagerDataSource dataSource = new DriverManagerDataSource();
        dataSource.setDriverClassName("org.hsqldb.jdbcDriver");
        dataSource.setUrl("jdbc:hsqldb:hsql://localhost:");
        dataSource.setUsername("sa");
        dataSource.setPassword("");
       
        // 新建一個ExecuteAStatement 對象
        ExecuteAStatement eas = new ExecuteAStatement();
        // 給執(zhí)行表達式的對象關(guān)聯(lián)數(shù)據(jù)源(也就是常說的注入, 通過 JavaBean 的 setXxx 方法關(guān)聯(lián)起來)
        eas.setDataSource(dataSource);
        // 執(zhí)行功能代碼
        eas.doExecute();
      }
    }

    這個代碼可以跑通, 就是普通的編程方式, 大家可以去看剛才介紹的文檔附近的詳細說明.

    那么如果用 Spring 來做, 代碼會變成這樣:
    ExecuteAStatement 類代碼保持不變, 多了個 beans.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"
     xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd">

     <bean id="userDAO" class="ExecuteAStatement">
      <property name="dataSource">
       <ref bean="myDataSource" />
      </property>
     </bean>

     <bean id="myDataSource"
      class="org.springframework.jdbc.datasource.DriverManagerDataSource">
      <property name="driverClassName">
          <value>org.hsqldb.jdbcDriver</value>
      </property>
      
      <property name="url">
          <value>jdbc:hsqldb:hsql://localhost:</value>
      </property>
      
      <property name="username">
          <value>sa</value>
      </property>
      
      <property name="password">
          <value></value>
      </property>      
     </bean>

    </beans>

    測試類:

    import org.springframework.context.ApplicationContext;
    import org.springframework.context.support.ClassPathXmlApplicationContext;

    public class Test {

     public static void main(String[] args) throws IOException {
      ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
      ExecuteAStatement eas =(ExecuteAStatement)context.getBean("userDAO");
          // 執(zhí)行功能代碼
          eas.doExecute();
     }
    }

    和上面的 TestTemplate 類相比, 就會發(fā)現(xiàn) new DriverManagerDataSource() 這個過程不用我們寫了, 運行的時候會發(fā)現(xiàn)一切都執(zhí)行的好好的, 也就是常說的 ExecuteAStatement 的 dataSource 這個屬性被注入了.

    那么這個過程到底該如何理解呢? Spring 是一個對象池, 可以簡化為一個 Map, 存多個主鍵和對象的映射. 那么 Spring 運行的過程中, 會根據(jù) beans.xml 一步步進行必要的解析工作:

    Map springEngine = new HashMap();

    OK, 解析到了
    <bean id="userDAO" class="ExecuteAStatement">, 發(fā)現(xiàn) bean 定義, 那就新建一個實例存到對象池里吧, 主鍵就是 userDAO, 值就是對象:
    ExecuteAStatement bean1 = new ExecuteAStatement();
    springEngine.put("userDAO", bean1);

    再往下執(zhí)行, 發(fā)現(xiàn) property 定義:
    <property name="dataSource">
    到了這里, 就知道應該調(diào)用 bean1.setDataSource(DataSource) 方法了. 可以接著執(zhí)行, 發(fā)現(xiàn)
    <ref bean="myDataSource" />, 哦, 這個方法的參數(shù)還沒有呢, 是個 bean 的引用, 好了, 要調(diào)用這個方法, 還是先 new 一個名字為 myDataSource 的 bean2 吧. 就跳到下面尋找 myDataSource 的定義, 找到了:
     <bean id="myDataSource"
      class="org.springframework.jdbc.datasource.DriverManagerDataSource">
      <property name="driverClassName">
          <value>org.hsqldb.jdbcDriver</value>
      </property>
      
      <property name="url">
          <value>jdbc:hsqldb:hsql://localhost:</value>
      </property>
      
      <property name="username">
          <value>sa</value>
      </property>
      
      <property name="password">
          <value></value>
      </property>      
     </bean>
    像以前一樣, 先實例化這個類, 然后看到 property 表情就調(diào)用對應的 setXxx() 這樣的方法, 相當于下面一段代碼:
        // 新建一個數(shù)據(jù)源對象
        DriverManagerDataSource bean2 = new DriverManagerDataSource();
        bean2.setDriverClassName("org.hsqldb.jdbcDriver");
        bean2.setUrl("jdbc:hsqldb:hsql://localhost:");
        bean2.setUsername("sa");
        bean2.setPassword("");
    不是還有個 bean 的 id 名字為 myDataSource 嘛, 那就把它存到對象池里面:

    springEngine.put("myDataSource", bean2);

    好了, 最后就是把他們兩個關(guān)聯(lián)起來了, 通過 ref 里指定的 bean id 名來關(guān)聯(lián)起來:

    // 省略類型轉(zhuǎn)換的代碼
    springEngine.get("userDAO").setDataSource(springEngine.get("myDataSource"));

    最后返回給用戶的就是一個對象池(一個 Map)了, 所以別人調(diào)用的時候, 就發(fā)現(xiàn) springEngine.get("userDAO") 回來的類的 dataSource 屬性已經(jīng)被實例化過了, 這些都是 Spring 幕后工作的代碼, 通過反射機制來實現(xiàn).

    所以最后寫代碼調(diào)用:
    context.getBean("userDAO") 的時候, 得到的是 ExecuteAStatement, 這時候還有一個 myDataSource, 也可以被調(diào)用:
    context.getBean("myDataSource"), 得到的是 DriverManagerDataSource.

    介紹的過程, 僅供參考. 歡迎大家交流更好的原理介紹文章.


     

    再轉(zhuǎn)一篇別人推薦的通俗易懂的說明, 非實現(xiàn)方面的:

    IoC就是Inversion of Control,控制反轉(zhuǎn)。在Java開發(fā)中,IoC意味著將你設計好的類交給系統(tǒng)去控制,而不是在你的類內(nèi)部控制。這稱為控制反轉(zhuǎn)。


    下面我們以幾個例子來說明什么是IoC


    假設我們要設計一個Girl和一個Boy類,其中Girl有kiss方法,即Girl想要Kiss一個Boy。那么,我們的問題是,Girl如何能夠認識這個Boy?


    在我們中國,常見的MM與GG的認識方式有以下幾種
    1 青梅竹馬; 2 親友介紹; 3 父母包辦
    那么哪一種才是最好呢?


    青梅竹馬:Girl從小就知道自己的Boy。


    public class Girl {
    void kiss(){
    Boy boy = new Boy();
    }
    }


    然而從開始就創(chuàng)建的Boy缺點就是無法在更換。并且要負責Boy的整個生命周期。如果我們的Girl想要換一個怎么辦?(嚴重不支持Girl經(jīng)常更換Boy,#_#)


    親友介紹:由中間人負責提供Boy來見面


    public class Girl {
    void kiss(){
    Boy boy = BoyFactory.createBoy();
    }
    }


    親友介紹,固然是好。如果不滿意,盡管另外換一個好了。但是,親友BoyFactory經(jīng)常是以Singleton的形式出現(xiàn),不然就是,存在于Globals,無處不在,無處不能。實在是太繁瑣了一點,不夠靈活。我為什么一定要這個親友摻和進來呢?為什么一定要付給她介紹費呢?萬一最好的朋友愛上了我的男朋友呢?


    父母包辦:一切交給父母,自己不用費吹灰之力,只需要等著Kiss就好了。


    public class Girl {
    void kiss(Boy boy){
    // kiss boy
    boy.kiss();
    }
    }


    Well,這是對Girl最好的方法,只要想辦法賄賂了Girl的父母,并把Boy交給他。那么我們就可以輕松的和Girl來Kiss了。看來幾千年傳統(tǒng)的父母之命還真是有用哦。至少Boy和Girl不用自己瞎忙乎了。


    這就是IOC,將對象的創(chuàng)建和獲取提取到外部。由外部容器提供需要的組件。


    我們知道好萊塢原則:“Do not call us, we will call you.” 意思就是,You, girlie, do not call the boy. We will feed you a boy。


    我們還應該知道依賴倒轉(zhuǎn)原則即 Dependence Inversion Princinple,DIP


    Eric Gamma說,要面向抽象編程。面向接口編程是面向?qū)ο蟮暮诵摹?


    組件應該分為兩部分,即 Service, 所提供功能的聲明 Implementation, Service的實現(xiàn)


    好處是:多實現(xiàn)可以任意切換,防止 “everything depends on everything” 問題.即具體依賴于具體。


    所以,我們的Boy應該是實現(xiàn)Kissable接口。這樣一旦Girl不想kiss可惡的Boy的話,還可以kiss可愛的kitten和慈祥的grandmother。
    二、IOC的type


    IoC的Type指的是Girl得到Boy的幾種不同方式。我們逐一來說明。


    IOC type 0:不用IOC
    public class Girl implements Servicable {
    private Kissable kissable;
    public Girl() {
    kissable = new Boy();
    }
    public void kissYourKissable() {
    kissable.kiss();
    }
    }

    Girl自己建立自己的Boy,很難更換,很難共享給別人,只能單獨使用,并負責完全的生命周期。


    IOC type 1,先看代碼:代碼


    public class Girl implements Servicable {

    Kissable kissable;

    public void service(ServiceManager mgr) {
    kissable = (Kissable) mgr.lookup(“kissable”);
    }

    public void kissYourKissable() {
    kissable.kiss();
    }

    }


    這種情況出現(xiàn)于Avalon Framework。一個組件實現(xiàn)了Servicable接口,就必須實現(xiàn)service方法,并傳入一個ServiceManager。其中會含有需要的其它組件。只需要在service方法中初始化需要的Boy。


    另外,J2EE中從Context取得對象也屬于type 1。它依賴于配置文件。


    IOC type 2:


    public class Girl {

    private Kissable kissable;

    public void setKissable(Kissable kissable) {
    this.kissable = kissable;
    }

    public void kissYourKissable() {
    kissable.kiss();
    }

    }


    Type 2出現(xiàn)于Spring Framework,是通過JavaBean的set方法來將需要的Boy傳遞給Girl。它必須依賴于配置文件。

    IOC type 3:


    public class Girl {

    private Kissable kissable;

    public Girl(Kissable kissable) {
    this.kissable = kissable;
    }

    public void kissYourKissable() {
    kissable.kiss();
    }

    }


    這就是PicoContainer的組件 。通過構(gòu)造函數(shù)傳遞Boy給Girl

    PicoContainer container = new DefaultPicoContainer();
    container.registerComponentImplementation(Boy.class);
    container.registerComponentImplementation(Girl.class);
    Girl girl = (Girl) container.getComponentInstance(Girl.class);
    girl.kissYourKissable();

    參考資料

    1 http://www.picocontainer.org/presentations/JavaPolis2003.ppt
    http://www.picocontainer.org/presentations/JavaPolis2003.pdf

    2 DIP, Robert C Martin, Bob大叔的優(yōu)秀論文
    http://www.objectmentor.com/resources/articles/dip.pdf

    3 Dependency Injection 依賴注射,Matrin Fowler對DIP的擴展
    http://www.martinfowler.com/articles/injection.html

    4 IOC框架

    PicoContainer 優(yōu)秀的IOC框架
    http://picocontainer.org/

    Avalon
    http://avalon.apache.org/

    Spring Framework
    http://www.springframework.org/

    HiveMind
    http://jakarta.apache.org/commons/hivemind

    ----

    IoC是一種模式
    IoC(Inversion of Control)中文譯為控制反轉(zhuǎn),目前Java社群中流行的各種輕量級容器的實現(xiàn)都是以IoC模式作為基礎(chǔ)的。控制反轉(zhuǎn)意味著在系統(tǒng)開發(fā)過程中,設計的類將交由容器去控制,而不是在類的內(nèi)部去控制,類與類之間的關(guān)系將交由容器處理,一個類在需要調(diào)用另一個類時,只要調(diào)用另一個類在容器中注冊的名字就可以得到這個類的實例,與傳統(tǒng)的編程方式有了很大的不同,”不用你找,我來提供給你”,這就是控制反轉(zhuǎn)的含義。Martin Fowler在他的一篇文章中給IoC起了一個更為直觀的名字:依賴注射DI(Dependency Injection)。下面先引入這個模式。

    在設計模式中,我們已經(jīng)習慣一種思維編程方式:Interface Driven Design 接口驅(qū)動,接口驅(qū)動有很多好處,可以提供不同靈活的子類實現(xiàn),增加代碼穩(wěn)定和健壯性等等,但是接口一定是需要實現(xiàn)的,也就是如下語句遲早要執(zhí)行:

    InterfaceA a = new InterfaceAImp();

    InterfaceAImp是接口InterfaceA的一個子類,IoC模式可以延緩接口的實現(xiàn),根據(jù)需要實現(xiàn),有個比喻:接口如同空的模型套,在必要時,需要向模型套注射石膏,這樣才能成為一個模型實體,因此,我們將人為控制接口的實現(xiàn)成為注射。IoC模式是解決調(diào)用者和被調(diào)用者之間的一種關(guān)系,上述InterfaceA實現(xiàn)語句表明當前是在調(diào)用被調(diào)用者InterfaceAImp,由于被調(diào)用者名稱寫入了調(diào)用者的代碼中,這產(chǎn)生了一個接口實現(xiàn)的原罪:彼此聯(lián)系,調(diào)用者和被調(diào)用者有緊密聯(lián)系,在UML中是用依賴 Dependency 表示。但是這種依賴在分離關(guān)注的思維下是不可忍耐的,必須切割,實現(xiàn)調(diào)用者和被調(diào)用者解耦,新的Ioc模式依賴注射 (Dependency Injection)模式由此產(chǎn)生了,也就是將依賴先剝離,然后在適當時候再注射進入。

    posted on 2007-08-13 13:17 冬天出走的豬 閱讀(3098) 評論(0)  編輯  收藏 所屬分類: spring
    主站蜘蛛池模板: 天天影院成人免费观看| 亚洲一区二区三区免费视频| 日本不卡高清中文字幕免费| 亚洲国产成人手机在线电影bd| 3d动漫精品啪啪一区二区免费| 亚洲制服中文字幕第一区| 99精品视频免费在线观看| 久久久久亚洲AV无码专区首JN| 4虎1515hh永久免费| 亚洲图片在线观看| 在线日本高清免费不卡| 亚洲福利视频网址| 男人的好免费观看在线视频| 亚洲狠狠婷婷综合久久蜜芽| 日本v片免费一区二区三区| 国产亚洲视频在线| 亚洲日韩精品一区二区三区无码| 大地资源中文在线观看免费版| 亚洲男人天堂av| 中国在线观看免费高清完整版| 亚洲精品日韩一区二区小说| 亚洲无线一二三四区手机| 日本免费一区二区三区| 久久久国产亚洲精品| 免费女人18毛片a级毛片视频| 两个人日本免费完整版在线观看1 两个人的视频www免费 | 一级毛片全部免费播放| va天堂va亚洲va影视中文字幕 | 国产三级在线免费观看| 亚洲精品成人av在线| 最新猫咪www免费人成| 一个人看的www免费在线视频| 久久国产亚洲观看| 99精品全国免费观看视频| 一个人看的www在线免费视频| 亚洲视频在线观看免费视频| 日本高清免费不卡视频| 国产啪精品视频网站免费尤物| 国产亚洲精品成人AA片| 国产亚洲成av片在线观看| 成人最新午夜免费视频|