Create type at runtime that inherits an abstract class and implements an interface

后端 未结 1 919
轮回少年
轮回少年 2020-12-24 09:54

Our architecture uses the Repository pattern extensively. We have an abstract base class for most of the repositories that implements some common functionality (e.g. get, lo

相关标签:
1条回答
  • 2020-12-24 10:24

    I was intrigued by the possibilities raised by your question so I did some investigation into how you would generate the repository proxies using either Castle’s DynamicProxy or the Reflection.Emit classes.

    Using the following repository and domain classes (I expanded the scenario to allow repositories to return strongly type collections):

    public interface IRepository<T>
    {
        IEnumerable<T> All { get; }
    }
    
    public abstract class Repository<T> : IRepository<T>
    {
        public IEnumerable<T> All
        {
            get
            {
                return new T[0];
            }
        }
    }
    
    public interface IFooRepository : IRepository<Foo>
    {
    }
    
    public class Foo
    {
    }
    

    To generate a proxy equivalent to

    public class FooRepository : Repository<Foo>, IFooRepository
    {
    }
    

    when using DynamicProxy:

    DefaultProxyBuilder proxyBuilder = new DefaultProxyBuilder();
    
    Type baseType = typeof(Repository<>).MakeGenericType(typeof(Foo));
    
    Type[] intefacesImplemented =  new Type[] { typeof(IFooRepository) };
    
    Type proxy = proxyBuilder.CreateClassProxyType(baseType, intefacesImplemented,  ProxyGenerationOptions.Default);
    

    When using Reflection.Emit:

    Type baseType = typeof(Repository<>).MakeGenericType(typeof(Foo));
    Type repositoryInteface = typeof(IFooRepository);
    
    AssemblyName asmName = new AssemblyName(
        string.Format("{0}_{1}", "tmpAsm", Guid.NewGuid().ToString("N"))
    );
    
    // create in memory assembly only
    AssemblyBuilder asmBuilder =
        AppDomain.CurrentDomain.DefineDynamicAssembly(asmName, AssemblyBuilderAccess.Run);
    
    ModuleBuilder moduleBuilder =
        asmBuilder.DefineDynamicModule("core");
    
    string proxyTypeName = string.Format("{0}_{1}", repositoryInteface.Name, Guid.NewGuid().ToString("N"));
    
    TypeBuilder typeBuilder = 
        moduleBuilder.DefineType(proxyTypeName);
    
    typeBuilder.AddInterfaceImplementation(repositoryInteface);
    typeBuilder.SetParent(baseType);
    
    Type proxy = typeBuilder.CreateType();
    

    You can then register them with your IOC container and use them as usual: (in this case Windsor):

    WindsorContainer container = new WindsorContainer();
    
    container.Register(Component.For<IFooRepository>().Forward(proxy));
    
    IFooRepository repository = container.Resolve<IFooRepository>();
    
    IEnumerable<Foo> allFoos = repository.All;
    

    Both Reflection.Emit and DynamicProxy can be setup up allow for the use of non-default constructors.

    If you’re interested there’s an excellent tutorial on DynamicProxy, while the documentation for the Reflection.Emit classes can be found here.

    0 讨论(0)
提交回复
热议问题