莊周夢蝶

          生活、程序、未來
             :: 首頁 ::  ::  :: 聚合  :: 管理

          singleton迷戀

          Posted on 2008-02-23 15:34 dennis 閱讀(2175) 評論(5)  編輯  收藏 所屬分類: 模式與架構
              自從知道singleton模式這把錘子是什么樣的之后,我就把很多小疙瘩也當成了釘子,時常想象偶頂著模式的光環揮舞著“萬能”錘子在代碼叢林中學習蘭博搏斗的光輝形象~~~~。昨天讀《重構與模式》的inline singleton一節,一句話點醒夢中人,singleton模式是“保證一個類僅有一個實例,并提供一個訪問它的全局訪問點”,原來——singleton也是全局變量啊。馬教主深刻地教育過我們:全局變量在被證明無害之前都是有害。偶大汗淋漓。看看迷戀singleton的幾種現象:
          1、僅僅在一個地方調用到了某個singleton實例,并且對這個實例的處理代碼也集中在這么一兩個地方,這樣的情況下你為什么要singleton?這里需要一個個全局訪問點嗎?我看你是為了singleton而singleton。

          2、我為了性能優化啊!singleton只有一個實例,減小了創建開銷。oh,我終于找到一個用singleton的充分理由了——性能。慢著,跟我讀高大師的名言:“不成熟的優化是萬惡之源”。你怎么知道singleton對象的重復創建是明顯影響了性能?現代jvm對“短命”對象的創建代價已經非常低了。不成熟的優化不僅可能是無效的,而且也給以后重構工作帶來了困難。除非有明顯數據證明(分析工具而來)某個對象的重復創建是對性能影響極大,否則所謂性能優化不能成為采用singleton模式的理由。

          3、有時候我們需要在系統的不同層次間傳遞一些共享信息,如果不采用singleton對象來提供這些共享信息,就得在調用的方法中重復地傳遞這些參數,這是個應用singleton模式的場景。但是,如果這些共享信息是可被修改的,或者說singleton對象不是無狀態的,如果還采用singleton模式,那么你就不得不在調用的方法中從single對象取出舊信息和存入新信息,這樣的重復代碼將遍布的到處都是,不僅僅引入了同步訪問的需要,而且出錯的風險大大提高。這種情況下你還將這些信息作為方法參數傳遞而不是采用singleton可能更為清晰和健壯。

              singleton不僅僅是“保證一個類僅有一個實例”(這僅僅是描述),更重要的是它是用來“提供全局訪問點”的(這才是它的功能),不要再迷戀這把錘子,好好利用這把錘子。

          題外話:腳本語言似乎更容易濫用全局變量,javascript里可以模擬命名空間,Ruby也可以模擬類似的機制。最近寫的一個比較大一點的Ruby腳本,用了幾個全局變量(都是數組)用于保存狀態數據,一開始沒有意識到這一點,導致對全局變量的訪問散落在好幾個腳本文件里,RDT下看起來紅通通的一片極其不爽。那么就重構吧——封裝數組重構,將對這些全局數組的訪問和修改操作統一到一個模塊,調用全局變量的地方都引用這個模塊,通過模塊去操作全局變量,代碼看起來清爽多了。



          評論

          # re: singleton迷戀  回復  更多評論   

          2008-02-23 16:05 by 魔域私服
          深有同感`

          # re: singleton迷戀  回復  更多評論   

          2008-02-23 16:46 by Jack.Wang
          QQ 11843121
          MSN(Email) wbjeasygo@163.com
          技術與管理群 47763528

          # re: singleton迷戀  回復  更多評論   

          2008-02-23 21:29 by Arbow
          哎呀,工作第一年的時候,用了N多singleton來做服務獲取,用起來便利,但是發現做單元測試的時候一個字慘啊

          # re: singleton迷戀  回復  更多評論   

          2008-02-23 21:44 by Matthewchen
          其實,全局變量,單例模式,和類的靜態成員是不一樣的。

          # re: singleton迷戀  回復  更多評論   

          2008-02-24 09:54 by dennis
          @Matthewchen
          貌似我沒說一樣
          主站蜘蛛池模板: 贵南县| 利津县| 梁平县| 海盐县| 鹰潭市| 南部县| 张家港市| 奇台县| 黄龙县| 福海县| 汽车| 潜江市| 陆良县| 浦县| 景洪市| 新源县| 巫溪县| 莱阳市| 富裕县| 吐鲁番市| 丰原市| 丹巴县| 铜鼓县| 铁岭市| 银川市| 平原县| 无锡市| 玉溪市| 永济市| 榕江县| 清河县| 象州县| 黔南| 青田县| 永清县| 绥芬河市| 吉安县| 渝北区| 剑川县| 琼结县| 祁连县|