單例模式(Winform窗體的實現)

来源:http://www.cnblogs.com/mointor/archive/2016/01/28/5166988.html
-Advertisement-
Play Games

在我的設計模式分類當中,我選擇單例模式作為我第一個要寫的設計模式,其一,單例模式簡單、容易理解讓人接受,其二,單例模式很常用,在實際的Winform窗體應用開發中能夠帶來更好的客戶體驗。 單例模式的核心是在應用程式的生命周期中只實例化一次當前類,讓整個應用整個應用程式中只擁有一個當前類實例化的對象,


  在我的設計模式分類當中,我選擇單例模式作為我第一個要寫的設計模式,其一,單例模式簡單、容易理解讓人接受,其二,單例模式很常用,在實際的Winform窗體應用開發中能夠帶來更好的客戶體驗。

  單例模式的核心是在應用程式的生命周期中只實例化一次當前類,讓整個應用整個應用程式中只擁有一個當前類實例化的對象,在Winform應用程式中,我們顯示窗體的方法有兩種方法:第一種,Show()方法,這種方法當點擊多次按鈕的時候會顯示多個當前的窗體,造成操作不便。第二種,ShowDialog()方法,這種方法顯示出窗體之後,我們只能對當前窗體進行操作直到這個窗體關閉之後,用戶體驗不好。那怎樣才能只產生一個窗體,而且不影響對其他窗體的操作?解決這樣的問題我們最簡單的想法就是如果每次我們單機按鈕都使用相同的實例化對象,就只能產生一個窗體了,再使用Show()方法顯示窗體就不會影響其他窗體的操作了。

  說了這麼多,讓我們看看怎麼在窗體中使用單例模式吧,核心參考代碼如下:

public partial class FrmSingleton : Form
    {
        private static FrmSingleton frm = null;
        private FrmSingleton()
        {
            InitializeComponent();
        }

        public static FrmSingleton CreateInstrance()
        {
            if (frm == null)
            {
                frm = new FrmSingleton();
            }
            return frm;
        }
    }
FrmSingleton

  從上面的代碼我們可以看出使用單例模式有三個重要的要點:

    (1)構造方法一定要定義成私有的(這樣做的好處就是我們只能在當前類裡面實例化一個對象,類外面不能實例化,外界想使用的話我們可以給它提供一個靜態方法供外界獲取)

    (2)定義一個私有的數據類型為當前類的變數(用於保證類的實例化對象的唯一性)

    (3)定義一個靜態的方法用於給外界提供當前類的實例化對象

  上面是單例模式使用當中應該註意的地方,定義好了之後,我們就要在主窗體中的button事件中調用它了,具體代碼如下:

private void button1_Click(object sender, EventArgs e)
        {
            FrmSingleton FrmSingleton = FrmSingleton.CreateInstrance();
            FrmSingleton.Show();
        }
button事件調用

  好了,一個簡單的單例模式的Demo就寫完了,我們運行一下應用程式,多次點擊按鈕,也只產生一個窗體,是不是很有成就感,但是這個時候一定不要激動太早,當我們關閉當前打開的窗體之後,再次單機按鈕則提示我們“無法訪問已經釋放的記憶體”,其實也很容易解釋,當我們關閉窗體時,C#預設的垃圾回收機制會回收我們的frm對象,但此時frm對象並不為null,當我們再次使用frm==null進行判斷時,結果是false,返回出去的是釋放記憶體的frm對象,所以造成這樣的結果,那麼我們怎麼做才能避免異常呢,最簡單的做法就是在判斷frm==null的地方添加一個或運算,判斷一下是否已經釋放,如果釋放了,我們也要進行再次實例化,修改我們剛纔的代碼,修改後的代碼如下:

public partial class FrmSingleton : Form
    {
        private static FrmSingleton frm = null;
        private FrmSingleton()
        {
            InitializeComponent();
        }

        public static FrmSingleton CreateInstrance()
        {
            if (frm == null ||frm.IsDisposed)
            {
                frm = new FrmSingleton();
            }
            return frm;
        }
    }
FrmSingleton

  這樣我們在運行代碼,就不會出現我們剛纔遇到的問題了,一個簡單的單例模式這樣才算結束,又可以愉快的玩耍了。

  那麼問題又來了,假如說我有很多的窗體都要使用單例模式,難道我都要把每個窗體的代碼都要修改成那個樣子嗎,加入窗體多起來的話,那這種方式就會變得很麻煩,那有沒有一種簡單的方式,能讓我們不管增加多少窗體都能很簡單的使用我們的單例模式呢?

  再次回憶我們構建一個單例模式的三個要點,我們發現,它們的格式基本上沒有什麼不同,除了裡面使用到了不同的類型,對於不同的類型,我們可以很容易想到用泛型代替,這樣一個泛型類就提取了出來,具體代碼如下:

public class GenericSingleton<T> where T : Form, new()
    {
        private static T t = null;
        public static T CreateInstrance()
        {
            if (null == t || t.IsDisposed)
            {
                t = new T();
            }
            return t;
        }
    }
GenericSingleton

  泛型代碼已經寫好了,怎麼在事件中調用呢,其實很簡單和上面的用法基本相同,具體調用如下:

private void Add_Click(object sender, EventArgs e)
        {
            FrmAdd FrmAdd = GenericSingleton<FrmAdd>.CreateInstrance();
            FrmAdd.Show();
        }
button事件調用

  運行程式,當我們單機按鈕的時候,依然滿足我們的要求,同時也為添加多個窗體使用單例模式的便捷性提供了可能。

  本篇文章單例模式設計模式至此,謝謝您收看我的博客。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1 public class Stack { 2 private int maxSize=2;//棧容量,初始為2,(用於表達式求值,操作數棧) 3 private int top=-1;//棧頂指針 4 private int[] data=new int[maxSize];//數據 5 //判空
  • foreach迴圈時動態往數組裡添加數據,有一次做項目中,foreach的時候需要動態往數組裡添加數據(我們這裡隨便舉個例子) 結果: 哎?奇了怪了,這說明foreach迴圈時可以動態的往數組裡添加數據,為什麼$arr的數據確實被添加上了,但是沒有被foreach迴圈出來呢?網上查找得知,forea
  • zookeeper結合PropertyPlaceholderConfigurer實現的統一配置組件,巧妙的應用了PropertyPlaceholderConfigurer搜索多種數據源的優勢,且對原有代碼沒有任務的侵入性。
  • 語法 awk [ -F re] [parameter...] ['pattern {action}' ] [-f progfile][in_file...] 獲得普通外部變數 [xingxing.dxx@30_28_6_20 ~]$ test='test code' [xingxing.dxx@30
  • 漢字元在IntelliJ的控制台輸出亂碼。編譯器在編譯的時候,把漢字元編譯成非UTF-8而引起亂碼。我是在做Jsoup解析的時候出現的錯誤,其實歸根結底確實編譯器的原因。 解決方法: 1.修改.idea/encoding.xml。將對應工程的編碼方式(如GBK)改為UTF-8; 2.如果是Maven
  • 1, Swift 修改導航欄顏色 self.navigationController?.navigationBar.barTintColor 2, Swift button 屬性設置時直接進行初始化 var leftButton : UIButton = UIButton(type: UIButto
  • 1、建立DLL動態庫 動態鏈接庫(DLL)是從C語言函數庫和Pascal庫單元的概念發展而來的。所有的C語言標準庫函數都存放在某一函數庫中。在鏈接應用程式的過程中,鏈接器從庫文件中拷貝程式調用的函數代碼,並把這些函數代碼添加到可執行文件中。這種方法同只把函數儲存在已編譯的OBJ文件中相比更有利於代碼
  • 註意:本章代碼將會建立在上一章的代碼基礎上,上一章鏈接《第八章 企業項目開發--分散式緩存memcached》 1、為什麼用Redis 1.1、為什麼用分散式緩存(或者說本地緩存存在的問題)? 見《第八章 企業項目開發--分散式緩存memcached》 1.2、有了memcached,為什麼還要用r
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...