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

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

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

    GHawk

    2006年1月9日 #

    建議方正要是勝訴的話,為開(kāi)源事業(yè)做點(diǎn)貢獻(xiàn)

    中國(guó)人民可以說(shuō)是被盜版和Windows慣壞了。在Linux默認(rèn)環(huán)境下的中文顯示至今慘不忍睹。
    看看現(xiàn)在一些主要的發(fā)行版本,默認(rèn)設(shè)置下的日語(yǔ)的顯示已經(jīng)相當(dāng)不錯(cuò)了。
    作為漢字發(fā)祥地且擁有眾多人口的中國(guó),實(shí)在是有些悲哀。
    文泉驛計(jì)劃正在為實(shí)現(xiàn)這個(gè)目標(biāo)努力。作為一個(gè)非盈利性組織,他們的貢獻(xiàn)的確值得贊賞。
    最近方正正在為字體的事情打官司,并提出了高額賠償?shù)囊蟆=ㄗh方正要是能夠勝訴的話,貢獻(xiàn)一些字體給開(kāi)源組織,為弘揚(yáng)漢語(yǔ)言文化多做點(diǎn)貢獻(xiàn)。

    posted @ 2007-08-23 15:01 GHawk 閱讀(651) | 評(píng)論 (1)編輯 收藏

    Maven2 用于度量和品質(zhì)保證的插件


    以上是一些常用的用于品質(zhì)管理的插件。默認(rèn)情況下都不用配置,相當(dāng)方便。如果需要手動(dòng)配置的話,根據(jù)網(wǎng)上的文檔也相當(dāng)容易配置。
    apache的maven plugin頁(yè)面: http://maven.apache.org/plugins/
    codehaus mojo 頁(yè)面: http://mojo.codehaus.org

    posted @ 2007-05-28 14:32 GHawk 閱讀(1560) | 評(píng)論 (1)編輯 收藏

    Scripting in Mustang 的一點(diǎn)啟發(fā)

    2006 Sun Techdays Shanghai 的第2天下午有一個(gè)名為《Java Scripting: One VM, Many Languages》的Session。

    Rags為大家展示了Mustang的一個(gè)新特性,Scripting in Java——腳本語(yǔ)言支持。

    通過(guò)加入腳本引擎的支持,就能夠在Java中解釋Javascript,python,ruby等諸多腳本語(yǔ)言。

    對(duì)于這個(gè)特性,想到的一個(gè)可能的應(yīng)用就是在annotation中寫(xiě)腳本語(yǔ)言,然后在代碼中用相應(yīng)的腳本語(yǔ)言引擎解釋執(zhí)行。
    保留到運(yùn)行時(shí)的annotation可以用實(shí)現(xiàn)aop的功能,使用非inline的腳本就可以更靈活地控制aspect的行為。

    比如:
    //inline scripting
    @ScriptBefore(script
    ="",language="javascript"?)
    public?void?foo()?{
    ???
    }

    //non-inline scripting
    @ScriptBefore(file
    ="scripts/logging.js",language="javascript")
    public?void?bar()?{

    }

    posted @ 2006-09-26 10:04 GHawk 閱讀(1327) | 評(píng)論 (3)編輯 收藏

    一個(gè)XPer提供的一些經(jīng)驗(yàn)

    前些天,和一位XPer進(jìn)行了一次愉快的談話。他向我講述了一些感覺(jué)很有效的實(shí)踐。

    關(guān)于過(guò)程和迭代
    他曾經(jīng)參與過(guò)的項(xiàng)目的迭代是以月為迭代單位的,但事實(shí)上每周都會(huì)重復(fù)一個(gè)簡(jiǎn)單的過(guò)程。
    在迭代過(guò)程中,他非常推崇Burn-Down Charts。這是一個(gè)Scrum的工具。通過(guò)Burn-Down Charts,能夠把過(guò)程中間的變化記錄下來(lái),使過(guò)程高度可視化。等到一次迭代完成,回顧一下所有的Burn-Down Charts就能作為改進(jìn)的判斷依據(jù)。
    KPT Meeting。所謂KPT Meeting就是 Keep-Prevent-Try metting。小組定期舉行KPT會(huì)議(基本上是每周一次)。在KTP會(huì)議上,通過(guò)頭腦風(fēng)暴的方式每個(gè)人(不是某幾個(gè)人)把各自認(rèn)為前一階段里做得好的方面寫(xiě)在Keep一欄里;做得不好的方面寫(xiě)在Prevent一欄里;希望嘗試的寫(xiě)在Try一欄里。然后大家對(duì)這些項(xiàng)目進(jìn)行評(píng)估和篩選。下一階段中,Keep的項(xiàng)目繼續(xù)保持,Prevent的項(xiàng)目應(yīng)該杜絕,Try的項(xiàng)目進(jìn)行嘗試。

    工具
    在開(kāi)展這些實(shí)踐的時(shí)候,交流比較頻繁。首推的工具是Mini white boardDC
    選擇Mini white board的原因并不是因?yàn)閹в?mini"聽(tīng)上去會(huì)像 Mini Cooper 或者 iPod mini 那么cool。因?yàn)橐粔KA3左右大小的白板非常適合個(gè)人或者結(jié)對(duì)使用,而且環(huán)保(省去了草稿紙)。雖然整個(gè)團(tuán)隊(duì)也有用于大規(guī)模交流的更大的白板,但那屬于“競(jìng)爭(zhēng)資源”,各自使用自己的白板更為方便。
    交流結(jié)果產(chǎn)生后,為了不花不必要的時(shí)間去做精美的文檔,一臺(tái)輕便的DC往往是最合適的選擇。當(dāng)然,如果足夠,手機(jī)上的照相功能也可以完成同樣的任務(wù)。相比偷拍街上的MM,這些電子產(chǎn)品能夠?qū)崿F(xiàn)更大的價(jià)值。

    關(guān)于結(jié)對(duì)
    每天進(jìn)行6小時(shí)的結(jié)對(duì)編程,分3次,每次2小時(shí)。每次和不同的成員組隊(duì)。在結(jié)隊(duì)的時(shí)候充分利用了上面提到的工具進(jìn)行交流。如果出現(xiàn)兩個(gè)人不能解決的問(wèn)題的時(shí)候,會(huì)立即向整個(gè)團(tuán)隊(duì)提出,這樣可能導(dǎo)致一次stand-up meeting。即使問(wèn)題不能馬上解決,至少也能確保每個(gè)人都知道這個(gè)問(wèn)題。

    posted @ 2006-08-24 15:45 GHawk 閱讀(1445) | 評(píng)論 (0)編輯 收藏

    關(guān)于locale的設(shè)定 (轉(zhuǎn))

    轉(zhuǎn)自:http://www.syxin.com/2006/03/localelocale.html

    關(guān)于locale的設(shè)定
    ?
    locale是國(guó)際化與本土化過(guò)程中的一個(gè)非常重要的概念,個(gè)人認(rèn)為,對(duì)于中文用戶來(lái)說(shuō),通常會(huì)涉及到的國(guó)際化或者本土化,大致包含三個(gè)方面:看中文,寫(xiě)中文,與window中文系統(tǒng)的兼容和通信。從實(shí)際經(jīng)驗(yàn)上看來(lái),locale的設(shè)定與看中文關(guān)系不大,但是與寫(xiě)中文,及window分區(qū)的掛載方式有很密切的關(guān)系。本人認(rèn)為就像一個(gè)純英文的Windows能夠?yàn)g覽中文,日文或者意大利文網(wǎng)頁(yè)一樣,你不需要設(shè)定locale就可以看中文。那么,為什么要設(shè)定locale呢?什么時(shí)候會(huì)用到locale呢?

    一、為什么要設(shè)定locale
    正如前面我所講的,設(shè)定locale與你能否瀏覽中文的網(wǎng)頁(yè)沒(méi)有直接的關(guān)系,即便你把locale設(shè)置成en_US.ISO-8859-1這樣一個(gè)標(biāo)準(zhǔn)的英文locale你照樣可以瀏覽中文的網(wǎng)頁(yè),只要你的系統(tǒng)里面有相應(yīng)的字符集(這個(gè)都不一定需要)和合適的字體(如simsun),瀏覽器就可以把網(wǎng)頁(yè)翻譯成中文給你看。具體的過(guò)程是網(wǎng)絡(luò)把網(wǎng)頁(yè)傳送到你的機(jī)器上之后,瀏覽器會(huì)判斷相應(yīng)的編碼的字符集,根據(jù)網(wǎng)頁(yè)采用的字符集,去字體庫(kù)里面找合適的字體,然后由文字渲染工具把相應(yīng)的文字在屏幕上顯示出來(lái)。

    ?
    在下文本人會(huì)偶爾把字符集比喻成密碼本,個(gè)人覺(jué)得對(duì)于一些東西比較容易理解,假如你不習(xí)慣的話,把全文copy到任何文本編輯器,用字符集替換密碼本即可。
    ?
    那有時(shí)候網(wǎng)頁(yè)顯示亂碼或者都是方框是怎么回事呢?個(gè)人認(rèn)為,顯示亂碼是因?yàn)樵O(shè)定的字符集不對(duì)(或者沒(méi)有相應(yīng)的字符集),例如網(wǎng)頁(yè)是用UTF-8編碼的,你非要用GB2312去看,而系統(tǒng)根據(jù)GB2312去找字體,然后在屏幕上顯示,當(dāng)然是一堆的亂碼,也就是說(shuō)你用一個(gè)錯(cuò)誤的密碼本去翻譯發(fā)給你的電報(bào),當(dāng)然內(nèi)容那叫一個(gè)亂;至于有些時(shí)候?yàn)g覽的網(wǎng)頁(yè)能顯示一部分漢字,但有很多的地方是方框,能夠顯示漢字說(shuō)明瀏覽器已經(jīng)正確的判斷出了網(wǎng)頁(yè)的編碼,并在字體庫(kù)里面找到了相應(yīng)的文字,但是并不是每個(gè)字體庫(kù)都包含某個(gè)字符集全部的字體的緣故,有些時(shí)候會(huì)顯示不完全,找一個(gè)比較全的支持較多字符集的字體就可以了。
    ?

    既然我能夠?yàn)g覽中文網(wǎng)頁(yè),那為什么我還要設(shè)定locale呢?
    ?
    其實(shí)你有沒(méi)有想過(guò)這么一個(gè)問(wèn)題,為什么gentoo官方論壇上中文論壇的網(wǎng)頁(yè)是用UTF-8編碼的(雖然大家一直強(qiáng)烈建議用GB2312編碼),但是新浪網(wǎng)就是用GB2312編碼的呢?而Xorg的官方網(wǎng)頁(yè)竟然是ISO-8859-15編碼的,我沒(méi)有設(shè)定這個(gè)locale怎么一樣的能瀏覽呢?這個(gè)問(wèn)題就像是你有所有的密碼本,不論某個(gè)網(wǎng)站是用什么字符集編碼的,你都可以用你手里的密碼本把他們翻譯過(guò)來(lái),但問(wèn)題是雖然你能瀏覽中文網(wǎng)頁(yè),但是在整個(gè)操作系統(tǒng)里面流動(dòng)的還是英文字符。所以,就像你能聽(tīng)懂英語(yǔ),也能聽(tīng)懂中文。
    最根本的問(wèn)題是:你不可以寫(xiě)中文。
    ?
    當(dāng)你決定要寫(xiě)什么東西的時(shí)候,首先要決定的一件事情是用那種語(yǔ)言,對(duì)于計(jì)算機(jī)來(lái)說(shuō)就是你要是用哪一種字符集,你就必須告訴你的linux系統(tǒng),你想用那一本密碼本去寫(xiě)你想要寫(xiě)的東西。知道為什么需要用GB2312字符集去瀏覽新浪了吧,因?yàn)樾吕说木W(wǎng)頁(yè)是用GB2312寫(xiě)的。
    ?
    為了讓你的Linux能夠輸入中文,就需要把系統(tǒng)的locale設(shè)定成中文的(嚴(yán)格說(shuō)來(lái)是locale中的語(yǔ)言類別LC_CTYPE ),例如zh_CN.GB2312、zh_CN.GB18030或者zh_CN.UTF-8。很多人都不明白這些古里古怪的表達(dá)方式。這個(gè)外星表達(dá)式規(guī)定了什么東西呢?這個(gè)問(wèn)題稍后詳述,現(xiàn)在只需要知道,這是locale的表達(dá)方式就可以了。
    ?
    二、到底什么是locale?
    locale這個(gè)單詞中文翻譯成地區(qū)或者地域,其實(shí)這個(gè)單詞包含的意義要寬泛很多。Locale是根據(jù)計(jì)算機(jī)用戶所使用的語(yǔ)言,所在國(guó)家或者地區(qū),以及當(dāng)?shù)氐奈幕瘋鹘y(tǒng)所定義的一個(gè)軟件運(yùn)行時(shí)的語(yǔ)言環(huán)境。
    ?
    這個(gè)用戶環(huán)境可以按照所涉及到的文化傳統(tǒng)的各個(gè)方面分成幾個(gè)大類,通常包括用戶所使用的語(yǔ)言符號(hào)及其分類(LC_CTYPE),數(shù)字(LC_NUMERIC),比較和排序習(xí)慣(LC_COLLATE),時(shí)間顯示格式(LC_TIME),貨幣單位(LC_MONETARY),信息主要是提示信息,錯(cuò)誤信息, 狀態(tài)信息, 標(biāo)題, 標(biāo)簽, 按鈕和菜單等(LC_MESSAGES),姓名書(shū)寫(xiě)方式(LC_NAME),地址書(shū)寫(xiě)方式(LC_ADDRESS),電話號(hào)碼書(shū)寫(xiě)方式(LC_TELEPHONE),度量衡表達(dá)方式(LC_MEASUREMENT),默認(rèn)紙張尺寸大小(LC_PAPER)和locale對(duì)自身包含信息的概述(LC_IDENTIFICATION)。
    ?
    所以說(shuō),locale就是某一個(gè)地域內(nèi)的人們的語(yǔ)言習(xí)慣和文化傳統(tǒng)和生活習(xí)慣。一個(gè)地區(qū)的locale就是根據(jù)這幾大類的習(xí)慣定義的,這些locale定義文件放在/usr/share/i18n/locales目錄下面,例如en_US, zh_CN and de_DE@euro都是locale的定義文件,這些文件都是用文本格式書(shū)寫(xiě)的,你可以用寫(xiě)字板打開(kāi),看看里邊的內(nèi)容,當(dāng)然出了有限的注釋以外,大部分東西可能你都看不懂,因?yàn)槭怯玫腢nicode的字符索引方式。
    ?
    對(duì)于de_DE@euro的一點(diǎn)說(shuō)明,@后邊是修正項(xiàng),也就是說(shuō)你可以看到兩個(gè)德國(guó)的locale:
    /usr/share/i18n/locales/de_DE@euro
    /usr/share/i18n/locales/de_DE
    打開(kāi)這兩個(gè)locale定義,你就會(huì)知道它們的差別在于de_DE@euro使用的是歐洲的排序、比較和縮進(jìn)習(xí)慣,而de_DE用的是德國(guó)的標(biāo)準(zhǔn)習(xí)慣。
    ?
    上面我們說(shuō)到了zh_CN.GB18030的前半部分,后半部分是什么呢?大部分Linux用戶都知道是系統(tǒng)采用的字符集。
    ?
    三、什么是字符集?
    字符集就是字符,尤其是非英語(yǔ)字符在系統(tǒng)內(nèi)的編碼方式,也就是通常所說(shuō)的內(nèi)碼,所有的字符集都放在/usr/share/i18n/charmaps,所有的字符集也都是用Unicode編號(hào)索引的。Unicode用統(tǒng)一的編號(hào)來(lái)索引目前已知的全部的符號(hào)。而字符集則是這些符號(hào)的編碼方式,或者說(shuō)是在網(wǎng)絡(luò)傳輸,計(jì)算機(jī)內(nèi)部通信的時(shí)候,對(duì)于不同字符的表達(dá)方式,Unicode是一個(gè)靜態(tài)的概念,字符集是一個(gè)動(dòng)態(tài)的概念,是每一個(gè)字符傳遞或傳輸?shù)木唧w形式。就像Unicode編號(hào)U59D0是代表姐姐的“姐”字,但是具體的這個(gè)字是用兩個(gè)字節(jié)表示,三個(gè)字節(jié),還是四個(gè)字節(jié)表示,是字符集的問(wèn)題。例如:UTF-8字符集就是目前流行的對(duì)字符的編碼方式,UTF-8用一個(gè)字節(jié)表示常用的拉丁字母,用兩個(gè)字節(jié)表示常用的符號(hào),包括常用的中文字符,用三個(gè)表示不常用的字符,用四個(gè)字節(jié)表示其他的古靈精怪的字符。而GB2312字符集就是用兩個(gè)字節(jié)表示所有的字符。需要提到一點(diǎn)的是Unicode除了用編號(hào)索引全部字符以外,本身是用四個(gè)字節(jié)存儲(chǔ)全部字符,這一點(diǎn)在談到掛載windows分區(qū)的時(shí)候是非常重要的一個(gè)概念。所以說(shuō)你也可以把Unicode看作是一種字符集(我不知道它和UTF-32的關(guān)系,反正UTF-32就是用四個(gè)字節(jié)表示所有的字符的),但是這樣表述符號(hào)是非常浪費(fèi)資源的,因?yàn)樵谟?jì)算機(jī)世界絕大部分時(shí)候用到的是一個(gè)字節(jié)就可以搞定的26個(gè)字母而已。所以才會(huì)有UTF-8,UTF-16等等,要不然大同世界多好,省了這許多麻煩。
    ?

    四、zh_CN.GB2312到底是在說(shuō)什么?
    Locale 是軟件在運(yùn)行時(shí)的語(yǔ)言環(huán)境, 它包括語(yǔ)言(Language), 地域 (Territory) 和字符集(Codeset)。一個(gè)locale的書(shū)寫(xiě)格式為: 語(yǔ)言[_地域[.字符集]]. 所以說(shuō)呢,locale總是和一定的字符集相聯(lián)系的。下面舉幾個(gè)例子:
    ?
    1、我說(shuō)中文,身處中華人民共和國(guó),使用國(guó)標(biāo)2312字符集來(lái)表達(dá)字符。
    zh_CN.GB2312=中文_中華人民共和國(guó)+國(guó)標(biāo)2312字符集。
    ?
    2、我說(shuō)中文,身處中華人民共和國(guó),使用國(guó)標(biāo)18030字符集來(lái)表達(dá)字符。
    zh_CN.GB18030=中文_中華人民共和國(guó)+國(guó)標(biāo)18030字符集。
    ?
    3、我說(shuō)中文,身處中華人民共和國(guó)臺(tái)灣省,使用國(guó)標(biāo)Big5字符集來(lái)表達(dá)字符。
    zh_TW.BIG5=中文_臺(tái)灣.大五碼字符集
    ?
    4、我說(shuō)英文,身處大不列顛,使用ISO-8859-1字符集來(lái)表達(dá)字符。
    en_GB.ISO-8859-1=英文_大不列顛.ISO-8859-1字符集
    ?
    5、我說(shuō)德語(yǔ),身處德國(guó),使用UTF-8字符集,習(xí)慣了歐洲風(fēng)格。
    de_DE.UTF-8@euro=德語(yǔ)_德國(guó).UTF-8字符集@按照歐洲習(xí)慣加以修正
    ?
    注意不是de_DE@euro.UTF-8,所以完全的locale表達(dá)方式是
    [語(yǔ)言[_地域][.字符集] [@修正值]
    ?
    生成的locale放在/usr/lib/locale/目錄中,并且每個(gè)locale都對(duì)應(yīng)一個(gè)文件夾,也就是說(shuō)創(chuàng)建了de_DE@euro.UTF-8 locale之后,就生成/usr/lib/locale/de_DE@euro.UTF-8/目錄,里面是具體的每個(gè)locale的內(nèi)容。
    ?
    五、怎樣去自定義locale
    在gentoo生成locale還是很容易的,首先要在USE里面加入userlocales支持,然后編輯locales.build文件,這個(gè)文件用來(lái)指示glibc生成locale文件。
    很多人不明白每一個(gè)條目是什么意思。 其實(shí)根據(jù)上面的說(shuō)明現(xiàn)在應(yīng)該很明確了。
    ?
    File: /etc/locales.build
    en_US/ISO-8859-1
    en_US.UTF-8/UTF-8
    ?
    zh_CN/GB18030
    zh_CN.GBK/GBK
    zh_CN.GB2312/GB2312
    zh_CN.UTF-8/UTF-8
    ?
    上面是我的locales.build文件,依次的說(shuō)明是這樣的:
    ?
    en_US/ISO-8859-1:生成名為en_US的locale,采用ISO-8859-1字符集,并且把這個(gè)locale作為英文_美國(guó)locale類的默認(rèn)值,其實(shí)它和en_US.ISO-8859-1/ISO-8859-1沒(méi)有任何區(qū)別。
    ?
    en_US.UTF-8/UTF-8:生成名為en_US.UTF-8的locale,采用UTF-8字符集。
    ?
    zh_CN/GB18030:生成名為zh_CN的locale,采用GB18030字符集,并且把這個(gè)locale作為中文_中國(guó)locale類的默認(rèn)值,其實(shí)它和zh_CN.GB18030/GB18030沒(méi)有任何區(qū)別。
    ?
    zh_CN.GBK/GBK:生成名為zh_CN.GBK的locale,采用GBK字符集。
    zh_CN.GB2312/GB2312:生成名為zh_CN.GB2312的locale,采用GB2312字符集。
    zh_CN.UTF-8/UTF-8:生成名為zh_CN.UTF-8的locale,采用UTF-8字符集。
    ?
    關(guān)于默認(rèn)locale,默認(rèn)locale可以簡(jiǎn)寫(xiě)成en_US或者zh_CN的形式,只是為了表達(dá)簡(jiǎn)單而已沒(méi)有特別的意義。
    ?
    Gentoo在locale定義的時(shí)候掩蓋了一些東西,也就是locale的生成工具:localedef。
    在編譯完glibc之后你可以用這個(gè)localedef 再補(bǔ)充一些locale,就會(huì)更加理解locale了。具體的可以看 localedef 的manpage。
    ?
    $localedef -f 字符集 -i locale定義文件 生成的locale的名稱
    例如
    $localedef -f UTF-8 -i zh_CN zh_CN.UTF-8
    ?
    上面的定義方法和在locales.build中設(shè)定zh_CN.UTF-8/UTF-8的結(jié)果是一樣一樣的。
    ?

    六、locale的五臟六腑
    ?
    剛剛生成了幾個(gè)locale,但是為了讓它們生效,必須告訴Linux系統(tǒng)使用那(幾)個(gè)locale。這就需要對(duì)locale的內(nèi)部機(jī)制有一點(diǎn)點(diǎn)的了解。在前面我已經(jīng)提到過(guò),locale把按照所涉及到的文化傳統(tǒng)的各個(gè)方面分成12個(gè)大類,這12個(gè)大類分別是:
    1、語(yǔ)言符號(hào)及其分類(LC_CTYPE)
    2、數(shù)字(LC_NUMERIC)
    3、比較和排序習(xí)慣(LC_COLLATE)
    4、時(shí)間顯示格式(LC_TIME)
    5、貨幣單位(LC_MONETARY)
    6、信息主要是提示信息,錯(cuò)誤信息, 狀態(tài)信息, 標(biāo)題, 標(biāo)簽, 按鈕和菜單等(LC_MESSAGES)
    7、姓名書(shū)寫(xiě)方式(LC_NAME)
    8、地址書(shū)寫(xiě)方式(LC_ADDRESS)
    9、電話號(hào)碼書(shū)寫(xiě)方式(LC_TELEPHONE)
    10、度量衡表達(dá)方式(LC_MEASUREMENT)
    11、默認(rèn)紙張尺寸大小(LC_PAPER)
    12、對(duì)locale自身包含信息的概述(LC_IDENTIFICATION)。
    ?
    其中,與中文輸入關(guān)系最密切的就是 LC_CTYPE, LC_CTYPE 規(guī)定了系統(tǒng)內(nèi)有效的字符以及這些字符的分類,諸如什么是大寫(xiě)字母,小寫(xiě)字母,大小寫(xiě)轉(zhuǎn)換,標(biāo)點(diǎn)符號(hào)、可打印字符和其他的字符屬性等方面。而locale定義zh_CN中最最重要的一項(xiàng)就是定義了漢字(Class “hanzi”)這一個(gè)大類,當(dāng)然也是用Unicode描述的,這就讓中文字符在Linux系統(tǒng)中成為合法的有效字符,而且不論它們是用什么字符集編碼的。
    ?
    LC_CTYPE
    % This is a copy of the "i18n" LC_CTYPE with the following modifications: - Additional classes: hanzi
    ?
    copy "i18n"
    ?
    class "hanzi"; /
    % <U3400>..<U4DBF>;/
    <U4E00>..<U9FA5>;/
    <UF92C>;<UF979>;<UF995>;<UF9E7>;<UF9F1>;<UFA0C>;<UFA0D>;<UFA0E>;/
    <UFA0F>;<UFA11>;<UFA13>;<UFA14>;<UFA18>;<UFA1F>;<UFA20>;<UFA21>;/
    <UFA23>;<UFA24>;<UFA27>;<UFA28>;<UFA29>
    END LC_CTYPE
    ?
    在en_US的locale定義中,并沒(méi)有定義漢字,所以漢字不是有效字符。所以如果要輸入中文必須使用支持中文的locale,也就是zh_XX,如zh_CN,zh_TW,zh_HK等等。
    ?
    另外非常重要的一點(diǎn)就是這些分類是彼此獨(dú)立的,也就是說(shuō)LC_CTYPE,LC_COLLATE和 LC_MESSAGES等等分類彼此之間是獨(dú)立的,可以根據(jù)用戶的需要設(shè)定成不同的值。這一點(diǎn)對(duì)很多用戶是有利的,甚至是必須的。例如,我就需要一個(gè)能夠輸入中文的英文環(huán)境,所以我可以把LC_CTYPE設(shè)定成zh_CN.GB18030,而其他所有的項(xiàng)都是en_US.UTF-8。
    ?

    七、怎樣設(shè)定locale呢?
    ?
    設(shè)定locale就是設(shè)定12大類的locale分類屬性,即 12個(gè)LC_*。除了這12個(gè)變量可以設(shè)定以外,為了簡(jiǎn)便起見(jiàn),還有兩個(gè)變量:LC_ALL和LANG。它們之間有一個(gè)優(yōu)先級(jí)的關(guān)系:
    LC_ALL>LC_*>LANG
    可以這么說(shuō),LC_ALL是最上級(jí)設(shè)定或者強(qiáng)制設(shè)定,而LANG是默認(rèn)設(shè)定值。
    1、如果你設(shè)定了LC_ALL=zh_CN.UTF-8,那么不管LC_*和LANG設(shè)定成什么值,它們都會(huì)被強(qiáng)制服從LC_ALL的設(shè)定,成為 zh_CN.UTF-8。
    2、假如你設(shè)定了LANG=zh_CN.UTF-8,而其他的LC_*=en_US.UTF-8,并且沒(méi)有設(shè)定LC_ALL的話,那么系統(tǒng)的locale設(shè)定以LC_*=en_US.UTF-8。
    3、假如你設(shè)定了LANG=zh_CN.UTF-8,而其他的LC_*,和LC_ALL均未設(shè)定的話,系統(tǒng)會(huì)將LC_*設(shè)定成默認(rèn)值,也就是LANG的值 zh_CN.UTF-8 。
    4、假如你設(shè)定了LANG=zh_CN.UTF-8,而其他的LC_CTYPE=en_US.UTF-8,其他的LC_*,和LC_ALL均未設(shè)定的話,那么系統(tǒng)的locale設(shè)定將是:LC_CTYPE=en_US.UTF-8,其余的 LC_COLLATE,LC_MESSAGES等等均會(huì)采用默認(rèn)值,也就是LANG的值,也就是LC_COLLATE=LC_MESSAGES=……= LC_PAPER=LANG=zh_CN.UTF-8。
    ?
    所以,locale是這樣設(shè)定的:
    1、如果你需要一個(gè)純中文的系統(tǒng)的話,設(shè)定LC_ALL= zh_CN.XXXX,或者LANG= zh_CN.XXXX都可以,當(dāng)然你可以兩個(gè)都設(shè)定,但正如上面所講,LC_ALL的值將覆蓋所有其他的locale設(shè)定,不要作無(wú)用功。
    2、如果你只想要一個(gè)可以輸入中文的環(huán)境,而保持菜單、標(biāo)題,系統(tǒng)信息等等為英文界面,那么只需要設(shè)定LC_CTYPE=zh_CN.XXXX,LANG=en_US.XXXX就可以了。這樣LC_CTYPE=zh_CN.XXXX,而LC_COLLATE=LC_MESSAGES=……= LC_PAPER=LANG=en_US.XXXX。
    3、假如你高興的話,可以把12個(gè)LC_*一一設(shè)定成你需要的值,打造一個(gè)古靈精怪的系統(tǒng):
    LC_CTYPE=zh_CN.GBK/GBK(使用中文編碼內(nèi)碼GBK字符集);
    LC_NUMERIC=en_GB.ISO-8859-1(使用大不列顛的數(shù)字系統(tǒng))
    LC_MEASUREMEN=de_DE@euro.ISO-8859-15(德國(guó)的度量衡使用ISO-8859-15字符集)
    羅馬的地址書(shū)寫(xiě)方式,美國(guó)的紙張?jiān)O(shè)定……。估計(jì)沒(méi)人這么干吧。
    4、假如你什么也不做的話,也就是LC_ALL,LANG和LC_*均不指定特定值的話,系統(tǒng)將采用POSIX作為lcoale,也就是C locale。

    posted @ 2006-08-16 10:45 GHawk 閱讀(757) | 評(píng)論 (0)編輯 收藏

    PostgreSQL的主機(jī)認(rèn)證配置

    轉(zhuǎn)自 http://www.linuxsir.org/bbs/showthread.php?t=32116

    pg_hba.conf 文件
    客戶端認(rèn)證是由 $PGDATA 目錄里的文件pg_hba.conf 控制的,也就是說(shuō), /usr/local/pgsql/data/pg_hba.conf. (HBA 的意思是 host-based authentication:基于主機(jī)的認(rèn)證.) 在initdb初始化數(shù)據(jù)區(qū)的時(shí)候,它會(huì) 安裝一個(gè)缺省的文件.

    文件 pg_hba.conf 的常用格式是一套記錄, 每行一條。空白行或者井號(hào)(“#”)開(kāi)頭的行被忽略。一條記錄 是由若干用空格和/或 tab 分隔的字段組成。

    每條記錄可以下面三種格式之一

    local database authentication-method [ authentication-option ]
    host database IP-address IP-mask authentication-method [ authentication-option ]
    hostssl database IP-address IP-mask authentication-method [ authentication-option ]

    各個(gè)字段的含義如下:

    local
    這條記錄適用于通過(guò) Unix 域套接字的聯(lián)接.

    host
    這條記錄適用于通過(guò) TCP/IP 網(wǎng)絡(luò)的聯(lián)接.請(qǐng)注意,除非服務(wù)器是 帶著 -i 選項(xiàng)或者等效的配置參數(shù)集啟動(dòng)的,否則 TCP/IP 聯(lián)接將完全被禁止掉.

    hostssl
    這條記錄適用于試圖建立在 TCP/IP 上的 SSL 之上的聯(lián)接. 要使用這個(gè)選項(xiàng),服務(wù)器必須帶著 SSL 支持編譯.而且在服務(wù)器啟動(dòng)的時(shí)候, 必須用 -l 選項(xiàng) 或等效的配置設(shè)置打開(kāi) SSL.

    database
    聲明記錄所適用的數(shù)據(jù)庫(kù)。值 all 表明該記錄應(yīng)用于所有數(shù)據(jù)庫(kù), 值 sameuser 表示于正在聯(lián)接的用戶同名的數(shù)據(jù)庫(kù)。 否則它就是某個(gè)具體的 Postgres 數(shù)據(jù)庫(kù)名字.

    IP address, IP mask
    這兩個(gè)字段以各主機(jī)的 IP 地址為基礎(chǔ), 控制一條 host 記錄應(yīng)用于哪個(gè)主機(jī). (當(dāng)然,IP 地址可能會(huì)被欺騙(spoofed),但是這個(gè)考慮 超過(guò)了 Postgres 的考慮范圍.) 準(zhǔn)確的邏輯是,對(duì)于匹配的記錄

    (actual-IP-address xor IP-address-field) and IP-mask-field
    必需為零.

    authentication method(認(rèn)證方法)
    聲明一個(gè)用戶在與該數(shù)據(jù)庫(kù)聯(lián)接的時(shí)候必須使用的認(rèn)證方法. 可能的選擇如下,詳細(xì)情況在 Section 4.2.


    trust
    無(wú)條件地允許聯(lián)接.這個(gè)方法允許任何有登錄客戶機(jī)權(quán)限的用戶以任意 Postgres 數(shù)據(jù)庫(kù)用戶身份進(jìn)行聯(lián)接.

    reject
    聯(lián)接無(wú)條件拒絕.常用于從組中“過(guò)濾”某些主機(jī).

    password
    要求客戶端在嘗試聯(lián)接的時(shí)候提供一個(gè)口令, 這個(gè)口令與為該用戶設(shè)置的口令必須匹配.

    在 password 關(guān)鍵字后面可以聲明一個(gè)可選的文件名. 這個(gè)文件包含一個(gè)用戶列表,列表記錄的是那些適用口令認(rèn)證記錄的用戶, 以及可選的候選口令.

    口令是以明文的方式在線路上傳輸?shù)模绻玫谋Wo(hù),請(qǐng)使用 crypt 方法.

    crypt
    類似 password 方法,但是口令是用一種簡(jiǎn)單的 口令對(duì)應(yīng)協(xié)議加密后在線路上傳送的.這么做在密碼學(xué)理論上是不安全的, 但可以防止偶然的線路偵聽(tīng).在 crypt 關(guān)鍵字后面 可以有一個(gè)文件,文件里包含適用口令認(rèn)證記錄的用戶列表.

    krb4
    用 Kerberos V4 認(rèn)證用戶.只有在進(jìn)行 TCP/IP 聯(lián)接的時(shí)候才能用. (譯注:Kerberos,"克爾波洛斯",故希臘神話冥王哈得斯的多頭看門(mén)狗. Kerberos 是 MIT 開(kāi)發(fā)出來(lái)的基與對(duì)稱加密算法的認(rèn)證協(xié)議和/或密鑰 交換方法.其特點(diǎn)是需要兩個(gè)不同用途的服務(wù)器,一個(gè)用于認(rèn)證身份, 一個(gè)用于通道兩端用戶的密鑰交換.同時(shí) Kerberos 對(duì)網(wǎng)絡(luò)時(shí)間同步 要求比較高,以防止回放攻擊,因此通常伴隨 NTP 服務(wù).)

    krb5
    用 Kerberos V5 認(rèn)證用戶.只有在進(jìn)行 TCP/IP 聯(lián)接的時(shí)候才能用. (譯注:Kerberos V5 是上面 V4 的改良,主要是不再依賴 DES 算法, 同時(shí)增加了一些新特性.)

    ident
    服務(wù)器將詢問(wèn)客戶機(jī)上的 ident 服務(wù)器以核實(shí)正在聯(lián)接的用戶身份. 然后 Postgres 核實(shí)該操作系統(tǒng)用戶是否被允許以其請(qǐng)求的數(shù)據(jù)庫(kù)用戶身份與數(shù)據(jù)庫(kù)聯(lián)接. 只有在使用 TCP/IP 聯(lián)接的時(shí)候才能用這個(gè)選項(xiàng). 跟在 ident 關(guān)鍵字后面的 authentication option 聲明一個(gè) ident map(身份映射), 該文件聲明那些操作系統(tǒng)用戶等效于數(shù)據(jù)庫(kù)用戶.見(jiàn)下文獲取詳細(xì)信息.


    authentication option(認(rèn)證選項(xiàng))
    這個(gè)字段根據(jù)不同的認(rèn)證方法(authentication method)有不同的 解釋.

    認(rèn)證時(shí)使用與聯(lián)接請(qǐng)求的客戶端 IP 地址和所要求的 數(shù)據(jù)庫(kù)名字匹配的第一條記錄. 請(qǐng)注意這里沒(méi)有 “fall-through(越過(guò))” 或者 “backup(備份)”:如果選定了一條記錄但認(rèn)證失敗, 那么將不會(huì)繼續(xù)考慮下面的記錄.如果沒(méi)有匹配的記錄,則拒絕訪問(wèn).

    在每次聯(lián)接的請(qǐng)求時(shí),文件 pg_hba.conf 都會(huì)被重新讀取.因此很容易就能在服務(wù)器運(yùn)行的時(shí)候修改訪問(wèn)權(quán)限.

    在 Example 4-1 里是 pg_hba.conf 的一個(gè)例子. 閱讀下文理解不同認(rèn)證方法的細(xì)節(jié).

    Example 4-1. 一個(gè) pg_hba.conf 文件的例子

    # TYPE DATABASE IP_ADDRESS MASK AUTHTYPE MAP

    # 允許在本機(jī)上的任何用戶以任何身份聯(lián)接任何數(shù)據(jù)庫(kù)
    # 但必須是通過(guò) IP 進(jìn)行聯(lián)接

    host all 127.0.0.1 255.255.255.255 trust

    # 同樣,但用的是 Unix-套接字聯(lián)接

    local all trust

    # 允許 IP 地址為 192.168.93.x 的任何主機(jī)與數(shù)據(jù)庫(kù)
    # "template1" 相連,用與他們?cè)谧约旱闹鳈C(jī)上相同 ident 的用戶名標(biāo)識(shí)他自己
    # (通常是他的 Unix 用戶名)

    host template1 192.168.93.0 255.255.255.0 ident sameuser

    # 允許來(lái)自主機(jī) 192.168.12.10 的用戶與 "template1" 數(shù)據(jù)庫(kù)聯(lián)接,
    # 只要該用戶提供了在 pg_shadow 里正確的口令.

    host template1 192.168.12.10 255.255.255.255 crypt

    # 如果前面沒(méi)有其它 "host" 行,那么下面兩行將拒絕所有來(lái)自
    # 192.168.54.1 的聯(lián)接請(qǐng)求 (因?yàn)榍懊娴挠涗浵绕ヅ?br /># 但是允許來(lái)自互聯(lián)網(wǎng)上其它任何地方有效的 Kerberos V5 認(rèn)證的聯(lián)接
    # 零掩碼表示不考慮主機(jī) IP 的任何位.因此它匹配任何主機(jī):

    host all 192.168.54.1 255.255.255.255 reject
    host all 0.0.0.0 0.0.0.0 krb5

    # 允許來(lái)自 192.168.x.x 的任何用戶與任意數(shù)據(jù)庫(kù)聯(lián)接,只要他們通過(guò) ident 檢查
    # 但如果 ident 說(shuō)該用戶是 "bryanh" 而他要求以 PostgreSQL 用戶 "guest1" 聯(lián)接,
    # 那么只有在 `pg_ident.conf' 里有 "omicron" 的映射,說(shuō) "bryanh" 允許以
    # "guest1" 進(jìn)行聯(lián)接時(shí)才真正可以進(jìn)行聯(lián)接.

    host all 192.168.0.0 255.255.0.0 ident omicron

    posted @ 2006-06-07 10:42 GHawk 閱讀(1214) | 評(píng)論 (0)編輯 收藏

    UP & XP之爭(zhēng),意義何在?(續(xù))

    雖然我沒(méi)能去參加BEA的活動(dòng),但是相關(guān)的資料已經(jīng)下載并且瀏覽過(guò)了,確實(shí)收獲不少。所以,對(duì)于莊兄的這些想法我很理解。

    相信不只你我,大部分的人都比較認(rèn)同敏捷化的過(guò)程,希望使過(guò)程變得敏捷。的確,這是個(gè)好東西,之前我也說(shuō)過(guò)“敏捷過(guò)程是三贏的”這樣的話。

    我所關(guān)心的問(wèn)題是“如何能夠用好XP?”。

    莊兄認(rèn)為“湯的味道,不需要什么過(guò)程控制”,我也會(huì)認(rèn)同。為什么?因?yàn)槟阄叶际侵袊?guó)人。大部分中國(guó)人不會(huì)認(rèn)為湯的味道需要什么過(guò)程控制。但是想想看,如果你在不同地方買(mǎi)到的肯德基炸雞味道各異;同一批次生產(chǎn)的同型號(hào)的汽車(chē)形狀各異;銀行里取出來(lái)的一疊百元大鈔大小不一,你不會(huì)覺(jué)得奇怪么或是有那么一點(diǎn)點(diǎn)憤怒么?

    西方人(甚至學(xué)習(xí)西方的日本人)對(duì)品質(zhì)的重視程度卻完全不同。他們不允許肯德基炸雞的味道有很大偏差(即便你覺(jué)得無(wú)所謂);“2毫米工程”不允許整車(chē)的總裝長(zhǎng)度發(fā)生2毫米以上的偏差(即便你覺(jué)得無(wú)所謂);百元大鈔……(我想誰(shuí)都不會(huì)無(wú)所謂)。

    所以,一切質(zhì)量都有標(biāo)準(zhǔn),一切標(biāo)準(zhǔn)都應(yīng)該被度量!這就是工程學(xué)的目標(biāo)之一,為了實(shí)現(xiàn)更嚴(yán)格的質(zhì)量標(biāo)準(zhǔn),就需要過(guò)程控制和度量。

    莊兄所說(shuō),用測(cè)試用例保證代碼的質(zhì)量其實(shí)還是采用了“測(cè)試用例”作為度量的標(biāo)準(zhǔn)。唯一的問(wèn)題是:“如何確保測(cè)試用例的質(zhì)量”。顯然,我們不能把一把不直的尺子度量出來(lái)的結(jié)果作為可靠的參考依據(jù)。怎么解決呢?“結(jié)對(duì)編程”么?嗯,這是一個(gè)不錯(cuò)的方式,那么最終該信賴誰(shuí)呢?是Pair中的A還是B呢?或者,是Leader么?那么又是誰(shuí)提出的要求呢?是老板么?還是客戶?政府?法規(guī)?市場(chǎng)?……問(wèn)題沒(méi)有終結(jié)了。

    不要學(xué)習(xí)哲學(xué)家的方法,提出一層又一層無(wú)法解決的問(wèn)題。我們是工程師,應(yīng)該試圖解決問(wèn)題才對(duì)!解決問(wèn)題的關(guān)鍵在于,XP同樣需要標(biāo)準(zhǔn)!為了制定標(biāo)準(zhǔn),必要的文檔是不可以少的。而且,標(biāo)準(zhǔn)本身的質(zhì)量是嚴(yán)苛的。因?yàn)椋鳛闃?biāo)準(zhǔn),他不可以含糊其辭、模棱兩可。在標(biāo)準(zhǔn)的基礎(chǔ)之上,我們才可以談什么TDD、Pair Programming之類的實(shí)踐。

    回到爭(zhēng)論的開(kāi)端。我引用了林先生的話“UP是正楷;XP是草書(shū)。要先學(xué)好UP才能學(xué)好XP,先學(xué)XP會(huì)亂套。”我對(duì)這句話的理解如下:這句話并沒(méi)有批判UP或是XP,只是指出了一個(gè)學(xué)習(xí)的順序。我認(rèn)為這句話是有實(shí)踐依據(jù)的,因?yàn)閁P強(qiáng)調(diào)的是一種經(jīng)典的工程方法。軟件工程本來(lái)就源于其他行業(yè)的工程實(shí)踐經(jīng)驗(yàn)。UP利用大量的文檔對(duì)開(kāi)發(fā)活動(dòng)進(jìn)行約束和記錄。正是這種重量級(jí)的過(guò)程規(guī)范了規(guī)范了從PM到Coder的所有活動(dòng),有問(wèn)題可以參照文檔,看看自己應(yīng)該怎么做。文檔也可以作為日后評(píng)估這個(gè)過(guò)程的依據(jù)。隨著整個(gè)團(tuán)隊(duì)和每個(gè)個(gè)人的經(jīng)驗(yàn)不斷積累,開(kāi)發(fā)活動(dòng)中的日常行為漸漸形成了一種職業(yè)習(xí)慣。然后可以通過(guò)對(duì)UP的配置,逐漸減少文檔的使用量,一些沒(méi)有必要的文檔就可以省去,更具團(tuán)隊(duì)的實(shí)際能力調(diào)整過(guò)程。UP是可配置的,不必要的文檔沒(méi)有存在的理由,這一點(diǎn)UP和XP沒(méi)有什么兩樣。當(dāng)然,隨著大家的職業(yè)習(xí)慣越來(lái)越好,經(jīng)驗(yàn)越來(lái)越豐富,個(gè)人和團(tuán)隊(duì)就可以采用更敏捷更輕便的過(guò)程,逐漸過(guò)渡到XP上去。

    反過(guò)來(lái),如果一開(kāi)始就沒(méi)有詳盡的文檔,很多活動(dòng)(比如設(shè)計(jì)、版本控制)往往會(huì)脫離控制,進(jìn)入一種無(wú)序的、混亂的狀態(tài)。沒(méi)有文檔可參考,就意味著很多問(wèn)題只能問(wèn)人,而不同人的回答可能各異,同一個(gè)人對(duì)同一個(gè)問(wèn)題的兩次回答也可能不同!當(dāng)然,如果整個(gè)團(tuán)隊(duì)的工程素養(yǎng)和個(gè)體的職業(yè)習(xí)慣都比較好的情況下可能不會(huì)發(fā)生類似的情況。但是這種工程素養(yǎng)和職業(yè)習(xí)慣從哪里來(lái),可能單靠的XP是不足以培養(yǎng)出來(lái)的。

    “UP是正楷;XP是草書(shū)。要先學(xué)好UP才能學(xué)好XP,先學(xué)XP會(huì)亂套。”這句話表明了UP和XP在一定程度上是存在沖突的,并且提出了一條路線去降低和避免這個(gè)沖突。

    再次需要強(qiáng)調(diào)的是莊兄所提到的“XP是一種思想”,這點(diǎn)我認(rèn)同。但是我認(rèn)為這個(gè)除了思想之外,還是一種“文化”。這種思想和文化也是出于軟件工程多年來(lái)的實(shí)踐,其中也不免有UP等其他過(guò)程。不能簡(jiǎn)單地認(rèn)為“我們只要吸取歷史的教訓(xùn),提出新的思想和文化就不會(huì)再犯同樣的錯(cuò)誤了。”很多時(shí)候歷史總是一次又一次地重演著。新的思想和文化如果不能被準(zhǔn)確地理解和運(yùn)用,它所帶來(lái)的可能仍然是它原本想解決的問(wèn)題。只有我們具備了引入這種文化的基礎(chǔ),才能把它變成自己的文化,否則這仍然是掛在嘴邊行于表面的一種不求精髓只求模仿的偽文化、偽思想。這一點(diǎn)對(duì)于UP和XP的實(shí)踐者來(lái)說(shuō)沒(méi)有什么兩樣。

    posted @ 2006-04-25 15:03 GHawk 閱讀(2050) | 評(píng)論 (4)編輯 收藏

    UP & XP之爭(zhēng),意義何在?

    不光是做軟件,凡是做產(chǎn)品,最后關(guān)注的總是產(chǎn)品的質(zhì)量

    舉個(gè)例子,比如你做一鍋湯:
    今天你狀態(tài)很好,做完后嘗了嘗,感覺(jué)很美味,你的家人嘗了以后也有同感,喝完后感覺(jué)心情舒暢、意猶未盡。
    隔了一個(gè)禮拜,你做同樣的湯給家里人喝。做完后你嘗了嘗,感覺(jué)依然美味,盼望著得到家人的賞識(shí),然而他們卻說(shuō)味道咸了點(diǎn)。你很奇怪,為什么同樣自己嘗過(guò)了,家里人卻感覺(jué)不一樣呢?是不是最近加班多了,休息不好,味覺(jué)不準(zhǔn)了?
    一個(gè)月過(guò)后,你要去國(guó)外出差,給家里請(qǐng)了個(gè)臨時(shí)保姆。一天,他也做了這么個(gè)湯,做完后,他也嘗了嘗,感覺(jué)口味很不錯(cuò),可是端上桌,家里人說(shuō)這湯太辣了。原來(lái)這保姆才從湖南老家出來(lái)不久……

    因此,只把焦點(diǎn)放在最后的產(chǎn)品上往往是不夠的。需要對(duì)“做湯的過(guò)程”加以控制。所以工程界會(huì)比較關(guān)注過(guò)程的管理,在軟件領(lǐng)域也稱作“軟件生命周期管理”。

    再來(lái)看看UP和XP。它們都屬于軟件過(guò)程,只不過(guò)各有特色。

    再拿剛才那個(gè)做湯的例子:
    大家都聽(tīng)說(shuō)過(guò)德國(guó)人的廚房像化學(xué)實(shí)驗(yàn)室,天平、計(jì)時(shí)器、量杯……裝備齊全,再配上精確的菜譜,嚴(yán)謹(jǐn)?shù)牡聡?guó)人能夠確保不用嘗那最后一口都做出口味基本一致的湯。
    換了中國(guó)人,大部分人都不會(huì)模仿德國(guó)人做菜的方式。解決方案很簡(jiǎn)單,讓你的太太和孩子都嘗那最后一口,再根據(jù)反饋調(diào)整幾次,同樣能做出全家人滿意的湯。

    這個(gè)例子也許不太貼切,但是可以聯(lián)想一下:德國(guó)人做湯傾向于UP;中國(guó)人做湯傾向于XP

    UP和XP最終目的都是為了保證產(chǎn)品的質(zhì)量,不同的是,兩個(gè)過(guò)程所強(qiáng)調(diào)的方法不同。我想,沒(méi)有人會(huì)說(shuō)“UP的目的在于變態(tài)地追求文檔的完美”、“UP是為了要程序員學(xué)會(huì)寫(xiě)各種各樣文檔”……之類的話。同時(shí),也沒(méi)人會(huì)說(shuō)“XP就是不要文檔只要代碼”、“XP就是要變態(tài)地追求完美的代碼”……這樣的話。

    這些不正確的看法,只是人們對(duì)于這兩種過(guò)程的誤解。或許是來(lái)自于開(kāi)發(fā)人員和項(xiàng)目經(jīng)理的那些“不堪回首的經(jīng)歷”。

    “UP害慘了整個(gè)軟件行業(yè),讓開(kāi)發(fā)人員沒(méi)完沒(méi)了地寫(xiě)文檔而忽略了代碼,XP才是王道”這樣的話,我不敢茍同,仍然有很多企業(yè)使用著UP這樣的重型軟件工程,就好比德國(guó)人依然喜歡把廚房弄得像個(gè)實(shí)驗(yàn)室。

    XP固然是個(gè)好東西。但是,不知道大多數(shù)人對(duì)于XP的熱衷是出于對(duì)XP文化的理解,還是國(guó)人慣有的“一窩蜂”似的行為。不曉得一個(gè)“能夠熟練閱讀代碼的Leader”是不是能夠真正運(yùn)用好XP,確保他的團(tuán)隊(duì)能夠盡可能少地出現(xiàn)"Over engineering"這種違背Agile精神的東西,或是能夠讓他的團(tuán)隊(duì)保證“每周只工作40小時(shí)”這樣的基本實(shí)踐?

    對(duì)于不同的技術(shù)和過(guò)程,應(yīng)該給予冷靜的分析和慎重的選擇。每個(gè)過(guò)程和技術(shù)都不能以“正確”或“不正確”來(lái)定性,只能以“合適”和“不合適”來(lái)定性。因?yàn)檎_或不正確是要嚴(yán)格證明的,而合適不合適是來(lái)源于工程實(shí)踐的結(jié)果。所以,COBOL依然在金融領(lǐng)域起著舉足輕重的作用,科學(xué)家們?nèi)圆煌麱ortran,匯編和C仍然健在……

    另外不得不提的是文化上的差異。為什么很多時(shí)候,我們學(xué)習(xí)國(guó)外的先進(jìn)技術(shù),購(gòu)買(mǎi)了整套生產(chǎn)線,引進(jìn)了全套圖紙,請(qǐng)國(guó)外專家做了詳細(xì)的全程化培訓(xùn),國(guó)人生產(chǎn)出的產(chǎn)品品質(zhì)依然不如國(guó)外原產(chǎn)的?這是每個(gè)中國(guó)人都應(yīng)該思考的問(wèn)題……

    ?

    posted @ 2006-04-23 18:28 GHawk 閱讀(1900) | 評(píng)論 (4)編輯 收藏

    對(duì)"UP是正楷,XP是草書(shū)"的反思

    “UP是正楷,XP是草書(shū)。先學(xué)好了UP,才能學(xué)好XP;先學(xué)XP再學(xué)UP就會(huì)亂套。?”

    老師曾這么說(shuō)。最近,對(duì)這句話有了深刻的體會(huì)。

    軟件過(guò)程是一個(gè)以人為中心的活動(dòng)。人是項(xiàng)目中最難確定和控制的因素。休息的質(zhì)量、情緒的起伏都會(huì)影響整個(gè)活動(dòng)。為了盡可能地約束這種個(gè)體的不確定行為和減少開(kāi)發(fā)過(guò)程中不必要的誤會(huì)。"UP"采用了大量的文檔來(lái)對(duì)整個(gè)開(kāi)發(fā)過(guò)程進(jìn)行控制。這些文檔主要分為以下幾類:

    • 計(jì)劃文檔——項(xiàng)目的開(kāi)發(fā)計(jì)劃、迭代計(jì)劃、測(cè)試計(jì)劃等。
    • 技術(shù)文檔——項(xiàng)目的設(shè)計(jì)文檔、某個(gè)操作的說(shuō)明文檔等。
    • 記錄文檔——日常的會(huì)議紀(jì)要、每日進(jìn)度反饋、評(píng)估報(bào)告等。

    文檔成了UP活動(dòng)的主要部分。在UP中,往往大量的資源用于文檔的制作。這些文檔的目的是為了盡可能減少不必要的溝通成本和誤會(huì),也為了在發(fā)生問(wèn)題的時(shí)候能夠盡快確定原因找到解決方法。

    而正是因?yàn)槿绱朔敝氐馁Y源消耗,導(dǎo)致真正的設(shè)計(jì)和代碼只占到了總開(kāi)銷(xiāo)的很少部分。這對(duì)很多人來(lái)說(shuō)不可理解,甚至覺(jué)得本末倒置。于是很多敏捷方法誕生了,最具代表性也是對(duì)UP思想最具顛覆性的就屬XP了。

    對(duì)外,XP以快速的反應(yīng)速度來(lái)響應(yīng)客戶的需求;對(duì)內(nèi),XP以高質(zhì)量的代碼和設(shè)計(jì)來(lái)確保盡可能不產(chǎn)生不必要的文檔和資源開(kāi)銷(xiāo)。

    從表面上看,在當(dāng)今,XP確實(shí)是一種非常理想的開(kāi)發(fā)過(guò)程。

    但是,從沒(méi)有過(guò)程到XP往往會(huì)非常失敗。這是為什么?問(wèn)題的關(guān)鍵還在于人。

    • 無(wú)過(guò)程-->UP -->XP

    UP利用文檔來(lái)約束和規(guī)范人們的開(kāi)發(fā)活動(dòng)。當(dāng)一個(gè)沒(méi)有經(jīng)驗(yàn)的團(tuán)隊(duì)經(jīng)歷UP后,就等于把性格各異、習(xí)慣差別不同的人統(tǒng)一成了“相對(duì)較一致的開(kāi)發(fā)人員”。

    他們有一致的編碼習(xí)慣,有共同的用語(yǔ),有嚴(yán)格的規(guī)則。隨著經(jīng)驗(yàn)的積累,這個(gè)團(tuán)隊(duì)間的默契越來(lái)越高。此時(shí),如果過(guò)程由UP向XP切換,付出的代價(jià)就會(huì)相對(duì)較低。

    • 無(wú)過(guò)程-->XP-->UP

    XP主張快速反應(yīng)。如果一個(gè)沒(méi)有經(jīng)驗(yàn)的團(tuán)隊(duì)在一開(kāi)始就嘗試XP,那么后果可能是慘痛的。因?yàn)橐粋€(gè)沒(méi)有經(jīng)驗(yàn)的團(tuán)隊(duì)其成員間的相互了解頗少,對(duì)于一件事,往往十個(gè)人有十種想法。當(dāng)缺少文檔約束時(shí),在以代碼和設(shè)計(jì)為中心的活動(dòng)中,成員之間往往因?yàn)樗降膮⒉畈积R導(dǎo)致無(wú)休止的討論甚至爭(zhēng)論,代碼被不必要地頻繁改動(dòng)。這是因?yàn)椋?font color="#ff0000">在團(tuán)隊(duì)建設(shè)早期,成員之間往往連最基本的尊重和信任都不存在。 這種無(wú)意義的活動(dòng)往往會(huì)嚴(yán)重影響項(xiàng)目的正常進(jìn)行。

    所以,學(xué)習(xí)和應(yīng)用過(guò)程不僅僅是個(gè)體的事,而是整個(gè)團(tuán)隊(duì)的事。只有當(dāng)團(tuán)隊(duì)采用嚴(yán)格文檔化的過(guò)程并且經(jīng)過(guò)磨合后,才能漸漸向輕量級(jí)的過(guò)程遷移,逐漸將不必要的文檔刪減掉,采用更靈活的過(guò)程。但是,此時(shí)并不是“沒(méi)有文檔”而是“心中有文檔”。

    posted @ 2006-03-01 16:25 GHawk 閱讀(1656) | 評(píng)論 (4)編輯 收藏

    加載Classpath中的文件(轉(zhuǎn))

       URL url = this.getClass().getResource("EJBConfig.xml");
            
    try {
                File xmlFile 
    = new File(URLDecoder.decode(url.getFile(),"UTF-8"));
                
    if(xmlFile.exists())
                    System.out.println(
    "OK");
            } 
    catch (UnsupportedEncodingException e) {
                e.printStackTrace();  
    //To change body of catch statement use File | Settings | File Templates.
            }

    posted @ 2006-01-19 22:07 GHawk 閱讀(764) | 評(píng)論 (0)編輯 收藏

    敏捷軟件開(kāi)發(fā) 讀書(shū)筆記 (4)——OO五大原則(3.LSP——里氏替換原則)

    OCP作為OO的高層原則,主張使用“抽象(Abstraction)”和“多態(tài)(Polymorphism)”將設(shè)計(jì)中的靜態(tài)結(jié)構(gòu)改為動(dòng)態(tài)結(jié)構(gòu),維持設(shè)計(jì)的封閉性。

    “抽象”是語(yǔ)言提供的功能。“多態(tài)”由繼承語(yǔ)義實(shí)現(xiàn)。

    如此,問(wèn)題產(chǎn)生了:“我們?nèi)绾稳ザ攘坷^承關(guān)系的質(zhì)量?”

    Liskov于1987年提出了一個(gè)關(guān)于繼承的原則“Inheritance should ensure that any property proved about supertype objects also holds for subtype objects.”——“繼承必須確保超類所擁有的性質(zhì)在子類中仍然成立。”也就是說(shuō),當(dāng)一個(gè)子類的實(shí)例應(yīng)該能夠替換任何其超類的實(shí)例時(shí),它們之間才具有is-A關(guān)系。

    該原則稱為L(zhǎng)iskov Substitution Principle——里氏替換原則。林先生在上課時(shí)風(fēng)趣地稱之為“老鼠的兒子會(huì)打洞”。^_^

    我們來(lái)研究一下LSP的實(shí)質(zhì)。學(xué)習(xí)OO的時(shí)候,我們知道,一個(gè)對(duì)象是一組狀態(tài)和一系列行為的組合體。狀態(tài)是對(duì)象的內(nèi)在特性,行為是對(duì)象的外在特性。LSP所表述的就是在同一個(gè)繼承體系中的對(duì)象應(yīng)該有共同的行為特征。

    這一點(diǎn)上,表明了OO的繼承與日常生活中的繼承的本質(zhì)區(qū)別。舉一個(gè)例子:生物學(xué)的分類體系中把企鵝歸屬為鳥(niǎo)類。我們模仿這個(gè)體系,設(shè)計(jì)出這樣的類和關(guān)系。

     lsp-fig1.jpg

    類“鳥(niǎo)”中有個(gè)方法fly,企鵝自然也繼承了這個(gè)方法,可是企鵝不能飛阿,于是,我們?cè)谄簌Z的類中覆蓋了fly方法,告訴方法的調(diào)用者:企鵝是不會(huì)飛的。這完全符合常理。但是,這違反了LSP,企鵝是鳥(niǎo)的子類,可是企鵝卻不能飛!需要注意的是,此處的“鳥(niǎo)”已經(jīng)不再是生物學(xué)中的鳥(niǎo)了,它是軟件中的一個(gè)類、一個(gè)抽象。

    有人會(huì)說(shuō),企鵝不能飛很正常啊,而且這樣編寫(xiě)代碼也能正常編譯,只要在使用這個(gè)類的客戶代碼中加一句判斷就行了。但是,這就是問(wèn)題所在!首先,客戶代碼和“企鵝”的代碼很有可能不是同時(shí)設(shè)計(jì)的,在當(dāng)今軟件外包一層又一層的開(kāi)發(fā)模式下,你甚至根本不知道兩個(gè)模塊的原產(chǎn)地是哪里,也就談不上去修改客戶代碼了。客戶程序很可能是遺留系統(tǒng)的一部分,很可能已經(jīng)不再維護(hù),如果因?yàn)樵O(shè)計(jì)出這么一個(gè)“企鵝”而導(dǎo)致必須修改客戶代碼,誰(shuí)應(yīng)該承擔(dān)這部分責(zé)任呢?(大概是上帝吧,誰(shuí)叫他讓“企鵝”不能飛的。^_^)“修改客戶代碼”直接違反了OCP,這就是OCP的重要性。違反LSP將使既有的設(shè)計(jì)不能封閉!

    修正后的設(shè)計(jì)如下:

     lsp-fig2.jpg

    但是,這就是LSP的全部了么?書(shū)中給了一個(gè)經(jīng)典的例子,這又是一個(gè)不符合常理的例子:正方形不是一個(gè)長(zhǎng)方形。這個(gè)悖論的詳細(xì)內(nèi)容能在網(wǎng)上找到,我就不多廢話了。

    LSP并沒(méi)有提供解決這個(gè)問(wèn)題的方案,而只是提出了這么一個(gè)問(wèn)題。

    于是,工程師們開(kāi)始關(guān)注如何確保對(duì)象的行為。1988年,B. Meyer提出了Design by Contract(契約式設(shè)計(jì))理論。DbC從形式化方法中借鑒了一套確保對(duì)象行為和自身狀態(tài)的方法,其基本概念很簡(jiǎn)單:

    1. 每個(gè)方法調(diào)用之前,該方法應(yīng)該校驗(yàn)傳入?yún)?shù)的正確性,只有正確才能執(zhí)行該方法,否則認(rèn)為調(diào)用方違反契約,不予執(zhí)行。這稱為前置條件(Pre-condition)。
    2. 一旦通過(guò)前置條件的校驗(yàn),方法必須執(zhí)行,并且必須確保執(zhí)行結(jié)果符合契約,這稱之為后置條件(Post-condition)。
    3. 對(duì)象本身有一套對(duì)自身狀態(tài)進(jìn)行校驗(yàn)的檢查條件,以確保該對(duì)象的本質(zhì)不發(fā)生改變,這稱之為不變式(Invariant)。

    以上是單個(gè)對(duì)象的約束條件。為了滿足LSP,當(dāng)存在繼承關(guān)系時(shí),子類中方法的前置條件必須與超類中被覆蓋的方法的前置條件相同或者更寬松;而子類中方法的后置條件必須與超類中被覆蓋的方法的后置條件相同或者更為嚴(yán)格。

    一些OO語(yǔ)言中的特性能夠說(shuō)明這一問(wèn)題:

    • 繼承并且覆蓋超類方法的時(shí)候,子類中的方法的可見(jiàn)性必須等于或者大于超類中的方法的可見(jiàn)性,子類中的方法所拋出的受檢異常只能是超類中對(duì)應(yīng)方法所拋出的受檢異常的子類。
      public class SuperClass{
          
      public void methodA() throws IOException{}
      }


      public class SubClassA extends SuperClass{
          
      //this overriding is illegal.
          private void methodA() throws Exception{}
      }


      public class SubClassB extends SuperClass{
          
      //this overriding is OK.
          public void methodA() throws FileNotFoundException{}
      }

    • 從Java5開(kāi)始,子類中的方法的返回值也可以是對(duì)應(yīng)的超類方法的返回值的子類。這叫做“協(xié)變”(Covariant)
      public class SuperClass {
          
      public Number caculate(){
              
      return null;
          }

      }


      public class SubClass extends SuperClass{
          
      //only compiles in Java 5 or later.
          public Integer caculate(){
              
      return null;
          }

      }

    可以看出,以上這些特性都非常好地遵循了LSP。但是DbC呢?很遺憾,主流的面向?qū)ο笳Z(yǔ)言(不論是動(dòng)態(tài)語(yǔ)言還是靜態(tài)語(yǔ)言)還沒(méi)有加入對(duì)DbC的支持。但是隨著AOP概念的產(chǎn)生,相信不久DbC也將成為OO語(yǔ)言的一個(gè)重要特性之一。

    一些題外話:

    前一陣子《敲響OO時(shí)代的喪鐘》和《喪鐘為誰(shuí)而鳴》兩篇文章引來(lái)了無(wú)數(shù)議論。其中提到了不少OO語(yǔ)言的不足。事實(shí)上,遵從LSP和OCP,不管是靜態(tài)類型還是動(dòng)態(tài)類型系統(tǒng),只要是OO的設(shè)計(jì),就應(yīng)該對(duì)對(duì)象的行為有嚴(yán)格的約束。這個(gè)約束并不僅僅體現(xiàn)在方法簽名上,而是這個(gè)具體行為的本身。這才是LSP和DbC的真諦。從這一點(diǎn)來(lái)說(shuō)并不能說(shuō)明“萬(wàn)事萬(wàn)物皆對(duì)象”的動(dòng)態(tài)語(yǔ)言和“C++,Java”這種“按接口編程”語(yǔ)言的優(yōu)劣,兩類語(yǔ)言都有待于改進(jìn)。莊兄對(duì)DJ的設(shè)想倒是開(kāi)始引入DbC的概念了。這一點(diǎn)還是非常值得期待的。^_^
    另外,接口的語(yǔ)義正被OCP、LSP、DbC這樣的概念不斷地強(qiáng)化,接口表達(dá)了對(duì)象行為之間的“契約”關(guān)系。而不是簡(jiǎn)單地作為一種實(shí)現(xiàn)多繼承的語(yǔ)法糖。

    posted @ 2006-01-18 18:12 GHawk 閱讀(3973) | 評(píng)論 (2)編輯 收藏

    敏捷軟件開(kāi)發(fā) 讀書(shū)筆記 (3)——OO五大原則(2.OCP——開(kāi)閉原則)

    開(kāi)閉原則很簡(jiǎn)單,一句話:“Closed for Modification; Open for Extension”——“對(duì)變更關(guān)閉;對(duì)擴(kuò)展開(kāi)放”。開(kāi)閉原則其實(shí)沒(méi)什么好講的,我將其歸結(jié)為一個(gè)高層次的設(shè)計(jì)總則。就這一點(diǎn)來(lái)講,OCP的地位應(yīng)該比SRP優(yōu)先。

    OCP的動(dòng)機(jī)很簡(jiǎn)單:軟件是變化的。不論是優(yōu)質(zhì)的設(shè)計(jì)還是低劣的設(shè)計(jì)都無(wú)法回避這一問(wèn)題。OCP說(shuō)明了軟件設(shè)計(jì)應(yīng)該盡可能地使架構(gòu)穩(wěn)定而又容易滿足不同的需求。

    為什么要OCP?答案也很簡(jiǎn)單——重用。

    “重用”,并不是什么軟件工程的專業(yè)詞匯,它是工程界所共用的詞匯。早在軟件出現(xiàn)前,工程師們就在實(shí)踐“重用”了。比如機(jī)械產(chǎn)品,通過(guò)零部件的組裝得到最終的能夠使用的工具。由于機(jī)械部件的設(shè)計(jì)和制造過(guò)程是極其復(fù)雜的,所以互換性是一個(gè)重要的特性。一輛車(chē)可以用不同的發(fā)動(dòng)機(jī)、不同的變速箱、不同的輪胎……很多東西我們直接買(mǎi)來(lái)裝上就可以了。這也是一個(gè)OCP的例子。(可能是由于我是搞機(jī)械出身的吧,所以就舉些機(jī)械方面的例子^_^)。

    如何在OO中引入OCP原則?把對(duì)實(shí)體的依賴改為對(duì)抽象的依賴就行了。下面的例子說(shuō)明了這個(gè)過(guò)程:

    05賽季的時(shí)候,一輛F1賽車(chē)有一臺(tái)V10引擎。但是到了06賽季,國(guó)際汽聯(lián)修改了規(guī)則,一輛F1賽車(chē)只能安裝一臺(tái)V8引擎。車(chē)隊(duì)很快投入了新賽車(chē)的研發(fā),不幸的是,從工程師那里得到消息,舊車(chē)身的設(shè)計(jì)不能夠裝進(jìn)新研發(fā)的引擎。我們不得不為新的引擎重新打造車(chē)身,于是一輛新的賽車(chē)誕生了。但是,麻煩的事接踵而來(lái),國(guó)際汽聯(lián)頻頻修改規(guī)則,搞得設(shè)計(jì)師在“賽車(chē)”上改了又改,最終變得不成樣子,只能把它廢棄。

    OCP-fig1.JPG

    為了能夠重用這輛昂貴的賽車(chē),工程師們提出了解決方案:首先,在車(chē)身的設(shè)計(jì)上預(yù)留出安裝引擎的位置和管線。然后,根據(jù)這些設(shè)計(jì)好的規(guī)范設(shè)計(jì)引擎(或是引擎的適配器)。于是,新的賽車(chē)設(shè)計(jì)方案就這樣誕生了。

     OCP-fig2.JPG

    顯然,通過(guò)重構(gòu),這里應(yīng)用的是一個(gè)典型的Bridge模式。這個(gè)實(shí)現(xiàn)的關(guān)鍵之處在于我們預(yù)先給引擎留出了位置!我們不必因?yàn)閷?duì)引擎的規(guī)則的頻頻變更而制造相當(dāng)多的車(chē)身,而是盡可能地沿用和改良現(xiàn)有的車(chē)身。
    說(shuō)到這里,想說(shuō)一說(shuō)OO設(shè)計(jì)的一個(gè)誤區(qū)。
    學(xué)習(xí)OO語(yǔ)言的時(shí)候,為了能夠說(shuō)明“繼承”(或者說(shuō)“is-a”)這個(gè)概念,教科書(shū)上經(jīng)常用實(shí)際生活中的例子來(lái)解釋。比如汽車(chē)是車(chē),電車(chē)是車(chē),F(xiàn)1賽車(chē)是汽車(chē),所以車(chē)是汽車(chē)、電車(chē)、F1賽車(chē)的上層抽象。這個(gè)例子并沒(méi)有錯(cuò)。問(wèn)題是,這樣的例子過(guò)于“形象”了!如果OO設(shè)計(jì)直接就可以將現(xiàn)實(shí)生活中的概念引用過(guò)來(lái),那也就不需要什么軟件工程師了!OO設(shè)計(jì)的關(guān)鍵概念是抽象。如果沒(méi)有抽象,那所有的軟件工程師的努力都是徒勞的。因?yàn)槿绻麤](méi)有抽象,我們只能去構(gòu)造世界中每一個(gè)對(duì)象。上面這個(gè)例子中,我們應(yīng)該看到“引擎”這個(gè)抽象的存在,因?yàn)檐?chē)隊(duì)的工程師們?yōu)樗A(yù)留了位置,為它制定了設(shè)計(jì)規(guī)范。
    上面這個(gè)設(shè)計(jì)也實(shí)現(xiàn)了后面要說(shuō)的DIP(依賴倒置原則)。但是請(qǐng)記住,OCP是OO設(shè)計(jì)原則中高層次的原則,其余的原則對(duì)OCP提供了不同程度的支持。為了實(shí)現(xiàn)OCP,我們會(huì)自覺(jué)或者不自覺(jué)地用到其它原則或是諸如Bridge、Decorator等設(shè)計(jì)模式。然而,對(duì)于一個(gè)應(yīng)用系統(tǒng)而言,實(shí)現(xiàn)OCP并不是設(shè)計(jì)目的,我們所希望的只是一個(gè)穩(wěn)定的架構(gòu)。所以對(duì)OCP的追求也應(yīng)該適可而止,不要陷入過(guò)渡設(shè)計(jì)。正如Martin本人所說(shuō):“No significant program can be 100% closed.”“Closure not complete but strategic”

    (下一篇就要講LSP了,我覺(jué)得這是意義最為重要的OO設(shè)計(jì)原則,它直指當(dāng)今主流OO語(yǔ)言的軟肋,點(diǎn)出了OO設(shè)計(jì)的精髓。)

    posted @ 2006-01-18 00:26 GHawk 閱讀(7576) | 評(píng)論 (7)編輯 收藏

    開(kāi)源 Java 測(cè)試框架(轉(zhuǎn))

    from  http://blog.csdn.net/wangyihust/archive/2006/01/02/568616.aspx

     JUnit   

    JUnit是由 Erich Gamma 和 Kent Beck 編寫(xiě)的一個(gè)回歸測(cè)試框架(regression testing framework)。Junit測(cè)試是程序員測(cè)試,即所謂白盒測(cè)試,因?yàn)槌绦騿T知道被測(cè)試的軟件如何(How)完成功能和完成什么樣(What)的功能。Junit是一套框架,繼承TestCase類,就可以用Junit進(jìn)行自動(dòng)測(cè)試了。

    http://www.junit.org/

     Cactus   

    Cactus是一個(gè)基于JUnit框架的簡(jiǎn)單測(cè)試框架,用來(lái)單元測(cè)試服務(wù)端Java代碼。Cactus框架的主要目標(biāo)是能夠單元測(cè)試服務(wù)端的使用Servlet對(duì)象的Java方法如HttpServletRequest,HttpServletResponse,HttpSession等。

    http://jakarta.apache.org/cactus/

     Abbot   

    Abbot是一個(gè)用來(lái)測(cè)試Java GUIs的框架。用簡(jiǎn)單的基于XML的腳本或者Java代碼,你就可以開(kāi)始一個(gè)GUI。

    http://abbot.sourceforge.net/

     JUnitPerf   

    Junitperf實(shí)際是junit的一個(gè)decorator,通過(guò)編寫(xiě)用于junitperf的單元測(cè)試,我們也可使測(cè)試過(guò)程自動(dòng)化。

    http://www.clarkware.com/software/JUnitPerf.html

     DbUnit   

    DbUnit是為數(shù)據(jù)庫(kù)驅(qū)動(dòng)的項(xiàng)目提供的一個(gè)對(duì)JUnit 的擴(kuò)展,除了提供一些常用功能,它可以將你的數(shù)據(jù)庫(kù)置于一個(gè)測(cè)試輪回之間的狀態(tài)。

    http://dbunit.sourceforge.net/

     Mockrunner   

    Mockrunner用在J2EE環(huán)境中進(jìn)行應(yīng)用程序的單元測(cè)試。它不僅支持Struts actions, servlets,過(guò)濾器和標(biāo)簽類還包括一個(gè)JDBC和一個(gè)JMS測(cè)試框架,可以用于測(cè)試基于EJB的應(yīng)用程序。

    http://mockrunner.sourceforge.net/index.html

     DBMonster   

    DBMonster是一個(gè)用生成隨機(jī)數(shù)據(jù)來(lái)測(cè)試SQL數(shù)據(jù)庫(kù)的壓力測(cè)試工具。

    http://dbmonster.kernelpanic.pl/

     MockEJB   

    MockEJB是一個(gè)不需要EJB容器就能運(yùn)行EJB并進(jìn)行測(cè)試的輕量級(jí)框架。

    http://mockejb.sourceforge.net/

     StrutsTestCase   

    StrutsTestCase 是Junit TestCase類的擴(kuò)展,提供基于Struts框架的代碼測(cè)試。StrutsTestCase同時(shí)提供Mock 對(duì)象方法和Cactus方法用來(lái)實(shí)際運(yùn)行Struts ActionServlet,你可以通過(guò)運(yùn)行servlet引擎來(lái)測(cè)試。因?yàn)镾trutsTestCase使用ActionServlet控制器來(lái)測(cè)試你的代碼,因此你不僅可以測(cè)試Action對(duì)象的實(shí)現(xiàn),而且可以測(cè)試mappings,from beans以及forwards聲明。StrutsTestCase不啟動(dòng)servlet容器來(lái)測(cè)試struts應(yīng)用程序(容器外測(cè)試)也屬于Mock對(duì)象測(cè)試,但是與EasyMock不同的是,EasyMock是提供了創(chuàng)建Mock對(duì)象的API,而StrutsTest則是專門(mén)負(fù)責(zé)測(cè)試Struts應(yīng)用程序的Mock對(duì)象測(cè)試框架。

    http://strutstestcase.sourceforge.net/

     JFCUnit   

    JFCUnit使得你能夠?yàn)镴ava偏移應(yīng)用程序編寫(xiě)測(cè)試?yán)印K鼮閺挠么a打開(kāi)的窗口上獲得句柄提供了支持;為在一個(gè)部件層次定位部件提供支持;為在部件中發(fā)起事件(例如按一個(gè)按鈕)以及以線程安全方式處理部件測(cè)試提供支持。

    http://jfcunit.sourceforge.net/

     JTestCase   

    JTestCase 使用XML文件來(lái)組織多測(cè)試案例數(shù)據(jù),聲明條件(操作和期望的結(jié)果),提供了一套易于使用的方法來(lái)檢索XML中的測(cè)試案例,按照數(shù)據(jù)文件的定義來(lái)聲明結(jié)果。

    http://jtestcase.sourceforge.net/

     SQLUnit   

    SQLUnit是一個(gè)單元測(cè)試框架,用于對(duì)數(shù)據(jù)庫(kù)存儲(chǔ)過(guò)程進(jìn)行回歸測(cè)試。用 Java/JUnit/XML開(kāi)發(fā)。

    http://sqlunit.sourceforge.net

     JTR   

    JTR (Java Test Runner)是一個(gè)開(kāi)源的基于反轉(zhuǎn)控制(IOC)的J2EE測(cè)試框架。它允許你構(gòu)建復(fù)雜的J2EE測(cè)試套件(Test Suites)并連到應(yīng)用服務(wù)器執(zhí)行測(cè)試,可以包括多個(gè)測(cè)試實(shí)例。JTR的licensed是GPL協(xié)議。

    http://jtrunner.sourceforge.net/

     Marathon   

    Marathon是一個(gè)針對(duì)使用Java/Swing開(kāi)發(fā)GUI應(yīng)用程序的測(cè)試框架,它由recorder, runner 和 editor組成,測(cè)試腳本是python代碼。Marathon的焦點(diǎn)是放在最終用戶的測(cè)試上。

    http://marathonman.sourceforge.net

     TestNG   

    TestNG是根據(jù)JUnit 和 NUnit思想而構(gòu)建的一個(gè)測(cè)試框架,但是TestNG增加了許多新的功能使得它變得更加強(qiáng)大與容易使用比如:
    *支持JSR 175注釋(JDK 1.4利用JavaDoc注釋同樣也支持)
    *靈活的Test配置
    *支持默認(rèn)的runtime和logging JDK功能
    *強(qiáng)大的執(zhí)行模型(不再TestSuite)
    *支持獨(dú)立的測(cè)試方法。

    http://testng.org/

     Surrogate Test framework   

    Surrogate Test framework是一個(gè)值得稱贊單元測(cè)試框架,特別適合于大型,復(fù)雜Java系統(tǒng)的單元測(cè)試。這個(gè)框架能與JUnit,MockEJB和各種支持模擬對(duì)象(mock object )的測(cè)試工具無(wú)縫給合。這個(gè)框架基于AspectJ技術(shù)。

    http://surrogate.sourceforge.net

     MockCreator   

    MockCreator可以為給定的interface或class生成模擬對(duì)象(Mock object)的源碼。

    http://mockcreator.sourceforge.net/

     jMock   

    jMock利用mock objects思想來(lái)對(duì)Java code進(jìn)行測(cè)試。jMock具有以下特點(diǎn):容易擴(kuò)展,讓你快速簡(jiǎn)單地定義mock objects,因此不必打破程序間的關(guān)聯(lián),讓你定義靈活的超越對(duì)象之間交互作用而帶來(lái)測(cè)試局限,減少你測(cè)試地脆弱性。

    http://www.jmock.org/

     EasyMock   

    EasyMock為Mock Objects提供接口并在JUnit測(cè)試中利用Java的proxy設(shè)計(jì)模式生成它們的實(shí)例。EasyMock最適合于測(cè)試驅(qū)動(dòng)開(kāi)發(fā)。

    http://www.easymock.org/

     The Grinder   

    The Grinder是一個(gè)負(fù)載測(cè)試框架。在BSD開(kāi)源協(xié)議下免費(fèi)使用。

    http://grinder.sourceforge.net/

     XMLUnit   

    XMLUnit不僅有Java版本的還有.Net版本的。Java開(kāi)發(fā)的XMLUnit提供了兩個(gè)JUnit 擴(kuò)展類XMLAssert和XMLTestCase,和一組支持的類。這些類可以用來(lái)比較兩張XML之間的不同之處,展示XML利用XSLT來(lái),校驗(yàn)XML,求得XPath表達(dá)式在XML中的值,遍歷XML中的某一節(jié)點(diǎn)利DOM展開(kāi)。

    http://xmlunit.sourceforge.net/

     Jameleon   

    Jameleon一個(gè)自動(dòng)化測(cè)試工具。它被用來(lái)測(cè)試各種各樣的應(yīng)用程序,所以它被設(shè)計(jì)成插件模式。為了使整個(gè)測(cè)試過(guò)程變得簡(jiǎn)單Jameleon提供了一個(gè)GUI,因此Jameleon實(shí)現(xiàn)了一個(gè)Swing 插件。

    http://jameleon.sourceforge.net/index.html

     J2MEUnit   

    J2MEUnit是應(yīng)用在J2ME應(yīng)用程序的一個(gè)單元測(cè)試框架。它基于JUnit。

    http://j2meunit.sourceforge.net/

     Jetif   

    Jetif是一個(gè)用純Java實(shí)現(xiàn)的回歸測(cè)試框架。它為Java程序單元測(cè)試以及功能測(cè)試提供了一個(gè)簡(jiǎn)單而且可 伸縮的架構(gòu),可以用于個(gè)人開(kāi)發(fā)或企業(yè)級(jí)開(kāi)發(fā)的測(cè)試。它容易使用,功能強(qiáng)大,而且擁有一些企業(yè)級(jí)測(cè)試的重要功能。Jetif來(lái)源于JUnit, JTestCase以及TestNG的啟發(fā),有幾個(gè)基本的概念直接來(lái)自于JUnit, 比如說(shuō)斷言機(jī)制,Test Listener的概念,因此從JUnit轉(zhuǎn)到Jetif是非常容易的。

    http://jetif.sourceforge.net/

     GroboUtils   

    GroboUtils使得擴(kuò)展Java測(cè)試變得可能。它包括用在Java不同方面測(cè)試的多個(gè)子項(xiàng)目。在GroboUtils中最常被到的工具是:多線程測(cè)試(multi-threaded tests),整體單元測(cè)試(hierarchial unit tests),代碼覆蓋工具(code coverage tool)。

    http://groboutils.sourceforge.net/

     Testare   

    TESTARE是用來(lái)簡(jiǎn)化分布式應(yīng)用程序(比如:在SERVLETS,JMS listeners, CORBA ORBs或RMI環(huán)境下)測(cè)試開(kāi)發(fā)過(guò)程的一個(gè)測(cè)試框架。

    https://testare.dev.java.net/

    posted @ 2006-01-10 10:41 GHawk 閱讀(1247) | 評(píng)論 (0)編輯 收藏

    敏捷軟件開(kāi)發(fā) 讀書(shū)筆記 (2)——OO五大原則(1.SRP 單一職責(zé)原則)

          一點(diǎn)說(shuō)明:OO的五大原則是指SRP、OCP、LSP、DIP、ISP。這五個(gè)原則是書(shū)中所提到的。除此之外,書(shū)中還提到一些高層次的原則用于組織高層的設(shè)計(jì)元素,這些放到下次再寫(xiě)。當(dāng)然,OO設(shè)計(jì)的原則可能不止這五個(gè),希望大家多提寶貴意見(jiàn),多多交流。

          在學(xué)習(xí)和使用OO設(shè)計(jì)的時(shí)候,我們應(yīng)該明白:OO的出現(xiàn)使得軟件工程師們能夠用更接近真實(shí)世界的方法描述軟件系統(tǒng)。然而,軟件畢竟是建立在抽象層次上的東西,再怎么接近真實(shí),也不能替代真實(shí)或被真實(shí)替代。

          OO設(shè)計(jì)的五大原則之間并不是相互孤立的。彼此間存在著一定關(guān)聯(lián),一個(gè)可以是另一個(gè)原則的加強(qiáng)或是基礎(chǔ)。違反其中的某一個(gè),可能同時(shí)違反了其余的原則。因此應(yīng)該把這些原則融會(huì)貫通,牢記在心!

    1. SRP(Single Responsibility Principle 單一職責(zé)原則)
          單一職責(zé)很容易理解,也很容易實(shí)現(xiàn)。所謂單一職責(zé),就是一個(gè)設(shè)計(jì)元素只做一件事。什么是“只做一件事”?簡(jiǎn)單說(shuō)就是少管閑事。現(xiàn)實(shí)中就是如此,如果要你專心做一件事情,任何人都有信心可以做得很出色。但如果,你整天被亂七八糟的事所累,還有心思和精力把每件事都作好么?
    fig-1.JPG
         “單一職責(zé)”就是要在設(shè)計(jì)中為每種職責(zé)設(shè)計(jì)一個(gè)類,彼此保持正交,互不干涉。這個(gè)雕塑(二重奏)就是正交的一個(gè)例子,鋼琴家和小提琴家各自演奏自己的樂(lè)譜,而結(jié)果就是一個(gè)和諧的交響樂(lè)。當(dāng)然,真實(shí)世界中,演奏小提琴和彈鋼琴的必須是兩個(gè)人,但是在軟件中,我們往往會(huì)把兩者甚至更多攪和到一起,很多時(shí)候只是為了方便或是最初設(shè)計(jì)的時(shí)候沒(méi)有想到。 

          這樣的例子在設(shè)計(jì)中很常見(jiàn),書(shū)中就給了一個(gè)很好的例子:調(diào)制解調(diào)器。這是一個(gè)調(diào)制解調(diào)器最基本的功能。但是這個(gè)類事實(shí)上完成了兩個(gè)職責(zé):連接的建立和中斷、數(shù)據(jù)的發(fā)送和接收。顯然,這違反了SRP。這樣做會(huì)有潛在的問(wèn)題:當(dāng)僅需要改變數(shù)據(jù)連接方式時(shí),必須修改Modem類,而修改Modem類的結(jié)果就是使得任何依賴Modem類的元素都需要重新編譯,不管它是不是用到了數(shù)據(jù)連接功能。解決的辦法,書(shū)中也已經(jīng)給出:重構(gòu)Modem類,從中抽出兩個(gè)接口,一個(gè)專門(mén)負(fù)責(zé)連接、另一個(gè)專門(mén)負(fù)責(zé)數(shù)據(jù)發(fā)送。依賴Modem類的元素也要做相應(yīng)的細(xì)化,根據(jù)職責(zé)的不同分別依賴不同的接口。最后由ModemImplementation類實(shí)現(xiàn)這兩個(gè)接口。
    fig-2.JPG

          從這個(gè)例子中,我們不難發(fā)現(xiàn),違反SRP通常是由于過(guò)于“真實(shí)”地設(shè)計(jì)了一個(gè)類所造成的。因此,解決辦法是往更高一層進(jìn)行抽象化提取,將對(duì)某個(gè)具體類的依賴改變?yōu)閷?duì)一組接口或抽象類的依賴。當(dāng)然,這個(gè)抽象化的提取應(yīng)該根據(jù)需要設(shè)計(jì),而不是盲目提取。比如剛才這個(gè)Modem的例子中,如果有必要,還可以把DataChannel抽象為DataSender和DataReceiver兩個(gè)接口。
     

    posted @ 2006-01-09 21:17 GHawk 閱讀(5557) | 評(píng)論 (5)編輯 收藏

    主站蜘蛛池模板: 免费在线观看日韩| 亚洲欧美乱色情图片| 国产99视频免费精品是看6| 日韩在线永久免费播放| 一级毛片**免费看试看20分钟| 亚洲视频一区在线观看| 区久久AAA片69亚洲| 免费国产成人高清在线观看麻豆| 国产1000部成人免费视频| GOGOGO免费观看国语| 相泽南亚洲一区二区在线播放| 亚洲一区中文字幕| 亚洲美女视频免费| 久久青青成人亚洲精品| 久久被窝电影亚洲爽爽爽| 国产精品亚洲精品日韩已方| 国产一区二区三区免费看| 免费观看的a级毛片的网站| 亚洲高清中文字幕免费| 蜜臀98精品国产免费观看| 国产激情免费视频在线观看| 91精品全国免费观看青青| 在线播放免费人成视频网站| 特a级免费高清黄色片| 特级毛片A级毛片免费播放| 理论亚洲区美一区二区三区| 校园亚洲春色另类小说合集| 337p日本欧洲亚洲大胆人人| 国产成人精品日本亚洲专| 亚洲制服在线观看| 亚洲一区在线视频| jlzzjlzz亚洲jzjzjz| 亚洲中文字幕久久精品蜜桃| 亚洲色www永久网站| 亚洲熟妇AV一区二区三区宅男| 亚洲 日韩 色 图网站| 亚洲欧洲国产综合AV无码久久| 亚洲AV永久无码精品一福利| 欧洲亚洲国产精华液| 黄色一级视频免费观看| 2022免费国产精品福利在线 |