看完這篇原型設計模式,還不會,請你吃瓜

来源:https://www.cnblogs.com/mhg215/archive/2023/02/20/17127080.html
-Advertisement-
Play Games

概述 使用原型實例指定創建對象的種類,並且通過拷貝這些原型創建新的對象。 在軟體系統開發中,有時候會遇到這樣的情況:我們需要用到多個相同實例,最簡單直接的方法是通過多次調用new方法來創建相同的實例。 student s=new student(); student s1=new student() ...


概述

使用原型實例指定創建對象的種類,並且通過拷貝這些原型創建新的對象。

在軟體系統開發中,有時候會遇到這樣的情況:我們需要用到多個相同實例,最簡單直接的方法是通過多次調用new方法來創建相同的實例。

student s=new student();
student s1=new student();
student s2=new student();

但是有一個問題,如果我用要使用的實例創建起來十分耗費資源,或者創建起來步驟比較繁瑣,上邊的代碼缺點就暴露出來了:耗費資源,每次創建實例都要重覆繁瑣的創建過程。原始模式可以很好地解決這個問題,使用原型模式我們不需要每次都new一個新的實例,而是通過拷貝原有的對象來完成創建,這樣我們就不需要在記憶體中創建多個對象,也不需要重覆複雜的創建過程了。下邊以克隆學生為例解釋原型模式的用法,代碼非常簡單。

C#通過this.MemberwiseClone()實現原型模式

    /// <summary>
    /// //原型抽象類
    /// </summary>
    public abstract class StudentPrototype
    {
        public string  Name { get;  }
        public StudentPrototype(string name)
        {
            Name=name;
        }
        public  abstract StudentPrototype Clone();
    }
    /// <summary>
    /// 學生類繼承原型抽象類 並重寫Clone();
    /// </summary>
    public class Student : StudentPrototype
    {
        public Student(string name) : base(name)
        {
        }

        public override StudentPrototype Clone()
        {
            //淺拷貝
            //值類型成員:全都複製一份,並且搞一份新的。
            //引用類型:只是複製其引用,並不複製其對象。
            return (StudentPrototype)this.MemberwiseClone();
        }
    }
Console.WriteLine("原型設計模式");
Student student=new Student("mhg");
Student student1=(Student)student.Clone();
Console.WriteLine(student.GetHashCode());
Console.WriteLine(student1.GetHashCode());
Console.WriteLine(student1.Name);

結論:實現該原型模式,第一需要定義一個抽象類,定義一個抽象方法;第二寫一個類繼承該抽象類。重寫抽象方法即可。重寫抽象方法的邏輯使用this.MemberwiseClone();

C#自己繼承ICloneable實現原型模式

  public class Teacher:ICloneable
    {
        public Teacher(string name)
        {
            Name=name;
        }
        public string Name { get; }

        public object Clone()
        {
           return this.MemberwiseClone();   
        }
    }
Console.WriteLine("C#自己繼承ICloneable");
Teacher teacher=new Teacher("mhg2");
Teacher teacher2=(Teacher)teacher.Clone();

  Console.WriteLine(teacher.GetHashCode());
  Console.WriteLine(teacher2.GetHashCode());
  Console.WriteLine(teacher2.Name);

結論:定義一個類繼承ICloneable,然後使用this.MemberwiseClone()實現,這種方式更簡單。

這裡需要註意一點:通過this.MemberWiseClone()獲取一個對象的實例屬於淺拷貝,對實例的簡單類型屬性進行全值拷貝(包含string類型),對複雜類型屬性只拷貝了引用。

下麵咱們驗證一下淺拷貝確實只對值類型成員全部複製了一份,搞成了一份新的,對於引用類型,只是複製了其引用,並不複製其對象。

我們還是繼續用Teacher這個類,在這個類裡面增加一個引用類型MyStudent,咱上代碼。

    public class Teacher : ICloneable
    {
        public string? Name { get; set; }
        public MyStudent? MyStudent { get; set; }
        public object Clone()
        {
            return this.MemberwiseClone();
        }
        public void Show()
        {
            Console.WriteLine($"Teacher:{Name}");
            Console.WriteLine($"MyStudent name:{MyStudent.Name}");
            Console.WriteLine($"MyStudent Age:{MyStudent.Age}");
        }
    }
   
    public class MyStudent
    {
        public string Name { get; set; }
        public string  Age { get; set; }
    }

看下執行結果

通過執行克隆了一份新對象,修改了Teacher.Mystudent.Name和Teacher.Mystudent.Age的值,其teacher對象Mystudent.Name和MyStudent.Age值也會發生變化,而修改了Teacher2.Name的值,其teacher對象的name卻沒有發生變化。也就驗證我們上面所說的,原型淺拷貝關於值類型全部複製一份,對於引用只複製其引用,這點特別重要,很多人搞不明白,多動手實踐一下。

那如果就上面的問題而言,我們現在既想對原型裡面的值類型複製一份新的,也想把引用類型複製一份新的對象,並不僅僅只是再複製其引用,該怎麼實現呢?

通過原型偽深拷貝實現

    public class Teacher : ICloneable
    {
        public string? Name { get; set; }
        public MyStudent? MyStudent { get; set; }
        public Teacher()
        {
            MyStudent=new MyStudent();    
        }
        private Teacher(MyStudent myStudent)
        {
            MyStudent=(MyStudent)myStudent.Clone();
        }
        public object Clone()
        {            
            //在創建新對象的時候把工作經驗這個引用類型也複製一份
            Teacher teacher1 = new Teacher(MyStudent)
            {
                Name = Name
            };
            return teacher1;

            //如果依然調用this.MemberwiseClone();引用類型,就永遠不可能被覆制一份新的
            //return this.MemberwiseClone(); //這種寫法只能拷貝值類型

        }
        public void Show(string objectName)
        {
            Console.WriteLine($"-------------{objectName}-start----------------");
            Console.WriteLine($"{objectName}:{Name}");
            Console.WriteLine($"MyStudent-name:{MyStudent.Name}");
            Console.WriteLine($"MyStudent-Age:{MyStudent.Age}");
            Console.WriteLine($"-------------{objectName}-end----------------\r\n");
        }
    }
   
    public class MyStudent:ICloneable
    {
        public string Name { get; set; }
        public string  Age { get; set; }

        public object Clone()
        {
            return this.MemberwiseClone();
        }
    }

來看看執行結果

通過上述代碼運行可以看出,teacher1、teacher2、teacher3幾個對象的創建...最後不僅把值類型全部複製了一份新的,引用類型也複製了一份對象,不再是複製其引用了。

目前這種原型創建還只是偽深拷貝,如果在MyStudent類中在出現一個引用類型,那麼就需要使用遞歸。這種方式顯而易見是有問題的,如果要真正的實現深拷貝,需要通過反射和序列化來實現.

總結

上述案例我們分別講了原型淺拷貝,原型偽深拷貝,如何實現真正的深拷貝,其實也很簡單,這次就不再往下寫了,文章寫短了沒人看,寫長了更沒人看!關於案例中的其他問題,有疑問,歡迎交流!

 

作者:realyrare

出處:https://www.cnblogs.com/mhg215/

聲援博主:如果您覺得文章對您有幫助,請點擊文章末尾的【關註我】吧!

別忘記點擊文章右下角的【推薦】支持一波。~~~///(^v^)\\\~~~ .

本文版權歸作者和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。

如果您有其他問題,也歡迎關註我下方的公眾號,可以聯繫我一起交流切磋!

碼雲:碼雲      github:github


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

-Advertisement-
Play Games
更多相關文章
  • 這是golang拾遺系列的第六篇。這個系列主要用來記錄一些平時不常見的知識點,偶爾也會實現些有意思的小功能,比如這篇。 golang拾遺系列目錄: golang拾遺:指針和介面 golang拾遺:為什麼我們需要泛型 golang拾遺:嵌入類型 golang拾遺:內置函數len的小知識 golang拾 ...
  • 原本也沒深究過這個,用的多了,完全憑藉經驗辦事,理論差的一塌糊塗,最近不流行那個openai,於是在偉大的人工智慧輔導下好好梳理一遍理論知識 初步理論認知 async 和 await 是 C# 語言中用於非同步編程的關鍵字,主要作用是讓代碼在等待非同步操作完成的時候繼續執行,從而達到不會阻塞線程的效果 ...
  • 如何優化線上WebAssembly WebAssembly部署使用 HTTPS : 為什麼?我可以通過一個案例查看 ,下麵我們會通過masa docs站點進行測試 打開 http://docs.masastack.com/blazor/getting-started/installation 網站 ...
  • 如何使用命名行指令去運行和打包.net6項目 前言 之前發佈了一個.net webApi入門項目,項目文章,在文章中我推薦的是Docker部署,只說明瞭如何打包,但是沒有說怎麼運行,考慮到很多人寫代碼不是用的Visual Studio。這裡講一下控制台怎麼去管理項目。 準備工作 安裝.net6 SD ...
  • 接前上一篇:平臺調用 (P/Invoke):DllImport特性說明 首先,我們知道C#和C/C++都是跨平臺的,但是原理上他們是不一樣的: C#首先編譯成一種中間語言(IL)的程式集,然後將這種程式集放到不同平臺下的解釋器裡面去執行,這就是說一次編譯到處運行 C/C++是針對不同的平臺直接編譯, ...
  • 1.說說.NET7中 _ViewImports文件的作用? 2.什麼是Razor頁面? 3.說說.NET5中 __ViewStart文件的作用? 4.如何在Razor頁面中實現數據模型綁定? 5.如何在Controller中註入service? 6.描述一下依賴註入後的服務生命周期? 7.說說ASP ...
  • 今年開年,最火的莫過於ChatGPT的相關討論,這個提供了非常強大的AI處理,並且整個平臺也提供了很多對應的API進行接入的處理,使得我們可以在各種程式上無縫接入AI的後端處理,從而實現智能AI的各種應用。ChatGPT的API可以在前端,以及一些後端進行API的接入,本篇隨筆主要介紹基於ChatG... ...
  • 一、什麼時候需要用到NuGet私有伺服器 很多公司中架構師會搭建一個統一的項目基礎架構模板,然後全部新項目都會拿這個基礎架構來開發新的項目,那架構中就會有很多的中間件,比喻公司內部的封裝好的Redis中間件,訪問資料庫的中間件,MQ中間件,小程式中間件等等。 現在很多項目都用這個模板開發了,然後其中 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...