将参数传递给需要用于泛型<t>?</t>的方法

时间:2013-06-07 15:22:31

标签: c# structuremap

我目前正在开发一个位于StructureMap之上的IoC容器抽象。这个想法是它也可以与其他IoC容器一起使用。

public interface IRegister
{
    IRegister RegisterType(Type serviceType, Type implementationType, params Argument[] arguments);
}

public abstract class ContainerBase : IRegister
{
    public abstract IRegister RegisterType(Type serviceType, Type implementationType, params Argument[] arguments);
}

public class StructureMapContainer : ContainerBase
{
    public StructureMapContainer(IContainer container)
    {
        Container = container;
    }

    public IContainer Container { get; private set; }

    public override IRegister RegisterType(Type serviceType, Type implementationType, params Argument[] arguments)
    {
        // StructureMap specific code
        Container.Configure(x =>
        {
            var instance = x.For(serviceType).Use(implementationType);
            arguments.ForEach(a => instance.CtorDependency<string>(a.Name).Is(a.Value));
        });

        return this;
    }
}

public class Argument
{       
    public Argument(string name, string value)
    {
        Name = name;
        Value = value;
    }

    public string Name { get; private set; }
    public string Value { get; private set; }
}

我可以通过执行以下操作来执行此代码:

//IContainer is a StructureMap type
IContainer container = new Container(); 
StructureMapContainer sm = new StructureMapContainer(container);

sm.RegisterType(typeof(IRepository), typeof(Repository));

当我尝试传入构造函数参数时出现问题。 StructureMap允许您根据需要流畅地链接许多CtorDependency调用,但需要构造函数参数名称,值和类型。所以我可以这样做:

sm.RegisterType(typeof(IRepository), typeof(Repository), new Argument("connectionString", "myConnection"), new Argument("timeOut", "360"));

此代码的工作原理是CtorDependency当前是string类型,两个参数也是字符串。

instance.CtorDependency<string>(a.Name).Is(a.Value)

如何将任何类型的多个构造函数参数传递给此方法?可能吗?任何帮助将不胜感激。

1 个答案:

答案 0 :(得分:2)

您可以使用动态的匿名对象传入您的参数。然后使用反射来检查属性。值可以是任何类型。

dynamic example = new
{
    a = 3,
    b = "hello"
};

foreach (PropertyInfo p in example.GetType().GetProperties())
{
    string key = p.Name;
    dynamic value = p.GetValue(example, null);
    Console.WriteLine("{0}: {1}", key, value);
}

此输出

  

a:3
  b:你好

根据需要进行修改。

修改

这很清楚。您的注册方法将采用动态参数。这将是某种东西。我没有StructureMap,但这应该让你朝着正确的方向前进。

public override IRegister RegisterType(Type serviceType, Type implementationType, dynamic arguments)
{
    // StructureMap specific code
    Container.Configure(x =>
    {
        var instance = x.For(serviceType).Use(implementationType);
        foreach (PropertyInfo p in arguments.GetType().GetProperties())
        {
            string key = p.Name;
            object value = p.GetValue(arguments, null);
            instance.CtorDependency<string>(key).Is(value));
        }
     });

    return this;
}

编辑零件解决方案:

好的,所以你希望能够为密钥传递任何数据类型。这更接近你所需要的吗?您可以按名称获取方法,通过调用MakeGenericMethod将其转换为封闭的泛型。然后你调用它。

var method = instance
    .GetType()
    .GetMethod("CtorDependency")
    .MakeGenericMethod(new Type[] { key.GetType() });
method.Invoke(instance, new object[] { key });