MEF(Managed Extensibility Framework) ile ilgili geçenlerde küçük bir soru ile karşılaştım. Burada da bahsetmek isterim. Ama önce MEF nedir, ne değildir hatırlamak isteyenlere önceki yazılara bir göz atmalarını tavsiye ederim…

  1. MEF ile esneklik kazanıyoruz…
  2. MEF’de “Part”lara kendi “metadata” bilgilerimizi nasıl ekleriz acaba?
  3. MEF’i basit bir WPF uygulaması ile daha iyi anlıyoruz…
  4. Managed Extensibility Framework(MEF)’de ki kataloglar…
  5. Asp.Net MVC Framework’de MEF ile Controller eklentileri…

Neyse gelelim, soruya… MEF alt yapısını kullanarak, uygulamalarızı genişletirken, “Part”ları [Export] özelliğini kullanarak belirtiyoruz. MEF’in olmazsa olmaz, en önemli özelliği aslında. Bununla ilgili olarak,  “Her geliştirilen “Part” bu şekilde [Export] özelliğine sahip olmalı mı,tek bir ortak yapı ile olmaz mı?” manasında bir soruydu gelen. Hemen bakalım…

MEF alt yapısı ile uygulamalarımızı genişletmek için kullandığımız parçaları mutlaka tanımlamamız lazım. MEF’deki kataloglar ve konteynırlar bu tanımlamalara göre oluşturduğumuz parçaları yaratıyor ve ayağa kalkmasını sağlıyor. Bildiğiniz üzere de [Export] bunu sağlayan özellik. Ama her parçaya bunu yazmak pek tercih edilen bir yöntem olmayabilir. Bu da aslında temel olarak iki sebepten olabilir. Bir tanesi MEF alt yapısını dışarısı ile paylaşmak istememek, bir diğeri de tüm koduna müdahele edemediğimiz uygulamaları genişletme ihtiyacı…

Öncelikle normalde nasıl yaptımızı hatırlayalım. Aşağıdaki kod örneği sanırım bunun için yeterli olacaktır.


    public interface IPlugin
    {
        void Execute();
    }

    [Export(typeof(IPlugin))]
    public class PluginA : IPlugin
    {

        public void Execute()
        {
            throw new NotImplementedException();
        }
    }

    [Export(typeof(IPlugin))]
    public class PluginB : IPlugin
    {

        public void Execute()
        {
            throw new NotImplementedException();
        }
    }

Burada her IPlugin arayüzünden geliştirilen, “Part”a, [Export] özelliğini yazmak yerine, BasePlugin şeklinde bir temel sınıf olup, “Part”ların o sınıftan türemesi istenebilir. Böylece “PlugX” şeklinde geliştirilen her sınıf, hem belli standart özellikleri sağlıyor olur, hem de [Export] özelliği o sınıf tarafından yönetilebilir. Ayrıca tüm koduna müdahele edemediğimiz, ama belli bir sınıftan türeyen sınıfların, MEF Part’ları olarak yorumlanmasını da istersek, bu belli sınıfa sadece bir özellik ekleyerek, yazılımcıdan “Part” kavramını soyutlayabiliriz.

MEF’de bunun için [InheritedExport] özelliği sunuyor.  Sadece bir sınıf ya da arayüze [InheritedExport] özelliği eklemek, bunlardan türetilen tüm sınıfların “Part” olarak yorumlanmasını sağlayacaktır. Bu sayede Export özelliği tek bir yerden yönetmek mümkün olacaktır.


    [InheritedExport(typeof(IPlugin))]
    public abstract class BasePlugin : IPlugin
    {

        public void Execute()
        {
            Process();
        }

        public abstract void Process();
    }

    public class PluginA : BasePlugin
    {

        public override void Process()
        {
            throw new NotImplementedException();
        }
    }

    public class PluginB : BasePlugin
    {

        public override void Process()
        {
            throw new NotImplementedException();
        }
    }

Bu basit ama bir o kadar da anlamlı özellik, kodun kirlenmesinin de önüne geçiyor diye düşünüyorum. MEF ile çalışmayı düşünen ya da çalışanlar için umarım faydalı olur.