Güzel bir konu ile tekrar merhaba arkadaşlar.
Bu makalemde sizlere nesnenin hallerini tutma ihtiyacı duyduğunuzda yani nesnenin farklı halleri arasında geçiş yapabilmeniz gerektiği durumlarda kullanabileceğiniz güzel bir tasarım deseni olan Memento Pattern’inin kullanımından bahsedeceğim.

Açıkcası GOF tasarım desenleri hakkındaki makalelerime başlangıcımı uzun zamandır düşünüyorum fakat bir türlü fırsat bulamamıştım, bir projem üzerinde kullanmam gerekince bu bana adı üstünde Hatırlayıcı(Memento) bana başlamamı hatırlattı. 🙂

Kendisi Behavioral tasarım kalıpları gurubunda olan bir tasarım desenidir. Yukarıda da bahsettiğimiz üzere nesnenin bazı özelliklerinin veya tamamını tutabilmemizi sağlamaktadır.

Hızlıca UML Diagramına bakmak gerekirse:

Originator: Bu sınıf durumu tutulacak olan nesnemiz oluyor, eski veya yeni halini tutmamızı sağlayacak metotlar burada yer alacak.
Memento: Bu sınıf ise, asıl nesnemizin istediğimiz alanları tutan sınıftır.
Caretaker: Geri dönüş adımlarımızı Memento tipinden tutacak olan sınıftır.

UML diagramını incelediğimize göre pekiştirmek için hemen basit bir örnek yapalım.

Web veya Windows tabanlı bir uygulama geliştirdiğimizi düşünelim. Uygulamamızda kullanıcının kendi ayarlarını yapabileceği bir Settings bölümü bulunsun. Memento desenini uygulayacağımız nokta ise: Get Default Settings tarzında bir buton koyacağız ve kullanıcı değiştirdiği ayarı orjinal haline geri döndürebilmesini sağlıyor olacağız.

Öncelikle Settings sınıfımızı oluşturalım ve bu bizim Originator‘umuzu temsil edecek yani ayarların tutulacağı sınıfımız.

    /// <summary>
    /// Originator - Ayarlarımızın tutulduğu sınıfımız. 
    /// </summary>
    public class Settings
    {
        // Kullanıcının güncelleyebileceği ayarlar propertyleri.
        public string UserName { get; set; }
        public string Password { get; set; }
        public string Email { get; set; }
        public bool RememberMe { get; set; }

        // Çağırıldığında Settings nesnemizin kopyasını oluşturacak olan metotumuz.
        public SettingsMemento Backup()
        {
            SettingsMemento settingsMemento = new SettingsMemento
            {
                UserName = this.UserName,
                Password = this.Password,
                Email = this.Email,
                RememberMe = this.RememberMe
            };
            return settingsMemento;
        }

        // Çağırıldığında daha önceden kopyasını aldığımız Memento sınıfındaki bilgileri Settings nesnemize geri atıyoruz.
        public void GetDefaultSettings(SettingsMemento settingsMemento)
        {
            this.UserName = settingsMemento.UserName;
            this.Password = settingsMemento.Password;
            this.Email = settingsMemento.Email;
            this.RememberMe = settingsMemento.RememberMe;
        }

        public override string ToString()
        {
            return string.Format("UserName: {0} Password: {1} Email: {2} RememberMe: {3}", this.UserName, this.Password, this.Email, this.RememberMe.ToString());
        }
    }

Settings sınıfımızı oluşturduğumuza göre şimdi Memento‘yu üstlenecek olan SettingsMemento sınıfımızı hazırlayalım.

    /// <summary>
    /// Memento - Asıl nesnemizin alanlarını tutan memento sınıfımız.
    /// </summary>
    public class SettingsMemento
    {
        public string UserName { get; set; }
        public string Password { get; set; }
        public string Email { get; set; }
        public bool RememberMe { get; set; }
    }

Şimdi SettingsCareTaker sınıfımızı oluşturarak geri dönüş adımlarımızı Memento tipinden tutacak olan sınıfımızı hazırlayalım.

    /// <summary>
    /// Caretaker - Geri dönüş adımlarımız burada tutulacak.
    /// </summary>
    public class SettingsCaretaker
    {
        public SettingsMemento Memento { get; set; }
    }

Ever her şey hazır olduğuna göre şimdi kullanımına bakalım.

            SettingsCaretaker settingsCaretaker = new SettingsCaretaker();

            Settings settings = new Settings();
            settings.UserName = "GokGokalp";
            settings.Password = "123456";
            settings.Email = "gok.gokalp@yahoo.com";
            settings.RememberMe = true;

            Console.WriteLine(settings.ToString() + Environment.NewLine);

            // Kopyasını alıyoruz.
            settingsCaretaker.Memento = settings.Backup();

            settings.Password = "654321";
            settings.RememberMe = false;

            Console.WriteLine(settings.ToString() + Environment.NewLine);

            // Caretaker üzerinde bulunan kopyasından orjinaline geri döndürüyoruz nesnemizi.
            settings.GetDefaultSettings(settingsCaretaker.Memento);

            Console.WriteLine(settings.ToString());
            Console.ReadLine();

İşte bu kadar basit. Umarım anlaşılması kolay bir örnek olmuştur.

Diğer makalelerimde görüşmek dileğiyle.

 

Gökhan Gökalp

Recent Posts

DevEx Series 03: Laying the Azure Focused Platform Foundation for an IDP with ASO and KRO

In the first two parts of this DevEx series, I tried to show how golden…

3 hafta ago

DevEx Series 02: From Catalog to Copilots. Boosting Backstage with MCP Server

In the first part of this DevEx series, I tried to explain Platform Engineering and…

5 ay ago

DevEx Series 01: Creating Golden Paths with Backstage, Developer Self-Service Without Losing Control

As an architect involved in platform engineering and DevEx transformation within a large-scale organization for…

6 ay ago

Event-Driven Architecture’larda Conditional Claim-Check Pattern’ı ile Event Boyut Sınırlarının Üstesinden Gelmek

{:en}In today’s technological age, we typically build our application solutions on event-driven architecture in order…

1 yıl ago

Containerized Uygulamaların Supply Chain’ini Güvence Altına Alarak Güvenlik Risklerini Azaltma (Güvenlik Taraması, SBOM’lar, Artifact’lerin İmzalanması ve Doğrulanması) – Bölüm 1

{:tr}Bildiğimiz gibi modern yazılım geliştirme ortamında containerization'ın benimsenmesi, uygulamaların oluşturulma ve dağıtılma şekillerini oldukça değiştirdi.…

2 yıl ago