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

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

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

    隨筆-12  評論-1  文章-0  trackbacks-0
     
    Software entities should be open for extension,but closed for modification.
    意思是說,一個軟件實(shí)體應(yīng)當(dāng)對擴(kuò)展開放,對修改關(guān)閉.也就是說,我們在設(shè)計一個模塊的時候,應(yīng)當(dāng)使這個模塊可以在不被修改的前提下被擴(kuò)展,換句話說就是,應(yīng)當(dāng)可以在不必修改源代碼的情況下改變這個模塊的行為.
    滿足OCP的設(shè)計給系統(tǒng)帶來兩個無可比擬的優(yōu)越性.
      1.通過擴(kuò)展已有的軟件系統(tǒng),可以提供新的行為,以滿足對軟件的新需求,使變化中的軟件系統(tǒng)有一定的適應(yīng)性和靈活性.
      2.已有的軟件模塊,特別是最重要的抽象層模塊不能再修改,這就使變化中的軟件系統(tǒng)有一定的穩(wěn)定性和延續(xù)性.
    例如:
      編程模式中的工廠模式的“工廠方法”支持OCP原則
    posted @ 2011-08-26 13:27 小熊寶貝的每一天 閱讀(160) | 評論 (0)編輯 收藏

    首先你要明白在java里面任何class都要裝載在虛擬機(jī)上才能運(yùn)行。這句話就是裝載類用的(和new 不一樣,要分清楚)。 

    至于什么時候用,你可以考慮一下這個問題,給你一個字符串變量,它代表一個類的包名和類名,你怎么實(shí)例化它?只有你提到的這個方法了,不過要再加一點(diǎn)。 
    A a = (A)Class.forName("pacage.A").newInstance(); 
    這和你 
    A a = new A(); 
    是一樣的效果。 

    關(guān)于補(bǔ)充的問題 
    答案是肯定的,jvm會執(zhí)行靜態(tài)代碼段,你要記住一個概念,靜態(tài)代碼是和class綁定的,class裝載成功就表示執(zhí)行了你的靜態(tài)代碼了。而且以后不會再走這段靜態(tài)代碼了。

    Class.forName(xxx.xx.xx) 返回的是一個類 
    Class.forName(xxx.xx.xx);的作用是要求JVM查找并加載指定的類,也就是說JVM會執(zhí)行該類的靜態(tài)代碼段

    動態(tài)加載和創(chuàng)建Class 對象,比如想根據(jù)用戶輸入的字符串來創(chuàng)建對象 
    String str = 用戶輸入的字符串 
    Class t = Class.forName(str); 
    t.newInstance();

     在初始化一個類,生成一個實(shí)例的時候,newInstance()方法和new關(guān)鍵字除了一個是方法,一個是關(guān)鍵字外,最主要有什么區(qū)別?它們的區(qū)別在于創(chuàng)建對象的方式不一樣,前者是使用類加載機(jī)制,后者是創(chuàng)建一個新類。那么為什么會有兩種創(chuàng)建對象方式?這主要考慮到軟件的可伸縮、可擴(kuò)展和可重用等軟件設(shè)計思想。 

    Java中工廠模式經(jīng)常使用newInstance()方法來創(chuàng)建對象,因此從為什么要使用工廠模式上可以找到具體答案。 例如: 
    class c = Class.forName(“Example”); 
    factory = (ExampleInterface)c.newInstance(); 

    其中ExampleInterface是Example的接口,可以寫成如下形式: 
    String className = "Example"; 
    class c = Class.forName(className); 
    factory = (ExampleInterface)c.newInstance(); 

    進(jìn)一步可以寫成如下形式: 
    String className = readfromXMlConfig;//從xml 配置文件中獲得字符串 
    class c = Class.forName(className); 
    factory = (ExampleInterface)c.newInstance(); 

    上面代碼已經(jīng)不存在Example的類名稱,它的優(yōu)點(diǎn)是,無論Example類怎么變化,上述代碼不變,甚至可以更換Example的兄弟類Example2 , Example3 , Example4……,只要他們繼承ExampleInterface就可以。 

    從JVM的角度看,我們使用關(guān)鍵字new創(chuàng)建一個類的時候,這個類可以沒有被加載。但是使用newInstance()方法的時候,就必須保證:1、這個類已經(jīng)加載;2、這個類已經(jīng)連接了。而完成上面兩個步驟的正是Class的靜態(tài)方法forName()所完成的,這個靜態(tài)方法調(diào)用了啟動類加載器,即加載 java API的那個加載器。 

    現(xiàn)在可以看出,newInstance()實(shí)際上是把new這個方式分解為兩步,即首先調(diào)用Class加載方法加載某個類,然后實(shí)例化。 這樣分步的好處是顯而易見的。我們可以在調(diào)用class的靜態(tài)加載方法forName時獲得更好的靈活性,提供給了一種降耦的手段。 

    最后用最簡單的描述來區(qū)分new關(guān)鍵字和newInstance()方法的區(qū)別: 
    newInstance: 弱類型。低效率。只能調(diào)用無參構(gòu)造。 
    new: 強(qiáng)類型。相對高效。能調(diào)用任何public構(gòu)造。

    下面內(nèi)容轉(zhuǎn)自
    http://blog.csdn.net/iceman1952/archive/2007/03/07/1523025.aspx
    介紹的是 forName() 和 ClassLoader 的 loadClass 方法。
    現(xiàn)在終于知道了為什么
    forName()是會執(zhí)行 static 語句,因?yàn)?/font>默認(rèn)情況它總是初始化這個被裝載的類。

    關(guān)于forName()方法
    這個方法總是返回要加載的類的Class類的實(shí)例
    1、forName(String className)單參數(shù)時, initialize=true
        a.總是使用當(dāng)前類裝載器(也就是裝載執(zhí)行forName()請求的類  的類裝載器)
        b.總是初始化這個被裝載的類(當(dāng)然也包括:裝載、連接、初始化)
    2、forName(String className, boolean initialize, ClassLoader loader)
        a.loader指定裝載參數(shù)類所用的類裝載器,如果null則用bootstrp裝載器。
        b.initialize=true時,肯定連接,而且初始化了;
        c.false時,絕對不會初始化,但是可能被連接了,但是這里有個例外,如果在調(diào)用這個forName()前,已經(jīng)被初始化了,那么返回的類型也肯定是被初始化的(當(dāng)然,這里也暗含著:被同一個loader所裝載的,而且這個類被初始化了)

    關(guān)于用戶自定義的類裝載器的loadClass()方法
    1、loadClass(String name)單參數(shù)時, resolve=false
        a.如果這個類已經(jīng)被這個類裝載器所裝載,那么,返回這個已經(jīng)被裝載的類型的Class的實(shí)例,否則,就用這個自定義的類裝載器來裝載這個class,這時不知道是否被連接。絕對不會被初始化
        b.這時唯一可以保證的是,這個類被裝載了。但是不知道這個類是不是被連接和初始化了
    2、loadClass(String name, boolean resolve)
        a.resolve=true時,則保證已經(jīng)裝載,而且已經(jīng)連接了。resolve=falses時,則僅僅是去裝載這個類,不關(guān)心是否連接了,所以此時可能被連接了,也可能沒有被連接




    終于明白為什么加載數(shù)據(jù)庫驅(qū)動只用Class.forName()了!!困擾了我2個小時!!希望我寫的這個東西對各位有所幫助。

      
       在Java開發(fā)特別是數(shù)據(jù)庫開發(fā)中,經(jīng)常會用到Class.forName( )這個方法。通過查詢Java Documentation我們會發(fā)現(xiàn)使用Class.forName( )靜態(tài)方法的目的是為了動態(tài)加載類。在加載完成后,一般還要調(diào)用Class下的newInstance( )靜態(tài)方法來實(shí)例化對象以便操作。因此,單單使用Class.forName( )是動態(tài)加載類是沒有用的,其最終目的是為了實(shí)例化對象。
       這里有必要提一下就是Class下的newInstance()和new有什么區(qū)別?,首先,newInstance( )是一個方法,而new是一個關(guān)鍵字,其次,Class下的newInstance()的使用有局限,因?yàn)樗蓪ο笾荒苷{(diào)用無參的構(gòu)造函數(shù),而使用 new關(guān)鍵字生成對象沒有這個限制。
       好,到此為止,我們總結(jié)如下:
       Class.forName("")返回的是類
       Class.forName("").newInstance()返回的是object
       有數(shù)據(jù)庫開發(fā)經(jīng)驗(yàn)朋友會發(fā)現(xiàn),為什么在我們加載數(shù)據(jù)庫驅(qū)動包的時候有的卻沒有調(diào)用newInstance( )方法呢?即有的jdbc連接數(shù)據(jù)庫的寫法里是Class.forName(xxx.xx.xx);而有一 些:Class.forName(xxx.xx.xx).newInstance(),為什么會有這兩種寫法呢?
       剛才提到,Class.forName("");的作用是要求JVM查找并加載指定的類,如果在類中有靜態(tài)初始化器的話,JVM必然會執(zhí)行該類的靜態(tài)代碼 段。而在JDBC規(guī)范中明確要求這個Driver類必須向DriverManager注冊自己,即任何一個JDBC Driver的 Driver類的代碼都必須類似如下:
      public class MyJDBCDriver implements Driver {
       static {
         DriverManager.registerDriver(new MyJDBCDriver());
      }
      }
    既然在靜態(tài)初始化器的中已經(jīng)進(jìn)行了注冊,所以我們在使用JDBC時只需要Class.forName(XXX.XXX);就可以了。

    posted @ 2011-08-26 10:12 小熊寶貝的每一天 閱讀(840) | 評論 (0)編輯 收藏
    僅列出標(biāo)題
    共2頁: 上一頁 1 2 
    主站蜘蛛池模板: 亚洲成人在线网站| 国产成人精品123区免费视频| 中文字幕日韩亚洲| 国产亚洲女在线线精品| 日韩毛片免费在线观看| 亚洲heyzo专区无码综合| 日韩免费视频播播| 黄色一级免费网站| 国产精品亚洲高清一区二区 | 亚洲综合伊人久久综合| 一级毛片一级毛片免费毛片| 亚洲片一区二区三区| 久久免费香蕉视频| 亚洲国产精彩中文乱码AV| 老汉精品免费AV在线播放| 亚洲国产综合精品| 免费黄色小视频网站| 免费大片av手机看片高清| 中文亚洲AV片不卡在线观看 | 性做久久久久久久免费看| 最新国产成人亚洲精品影院| 国产高清免费在线| 国产性生大片免费观看性 | 免费国产成人高清在线观看麻豆| 国产高清对白在线观看免费91 | 精品丝袜国产自在线拍亚洲| 色吊丝最新永久免费观看网站| 老湿机一区午夜精品免费福利| 国产亚洲高清不卡在线观看| 18禁止看的免费污网站| 天天综合亚洲色在线精品| 亚洲无线码在线一区观看| 日本人的色道免费网站| 国产亚洲蜜芽精品久久| 亚洲天天做日日做天天看| 天堂在线免费观看中文版| 中文字幕乱理片免费完整的| 国产精品亚洲综合久久| 亚洲日本乱码在线观看| 成人免费无遮挡无码黄漫视频| 中文字幕在线观看免费|