我正在尝试使用HangFire IoC为ServiceStack编写JobActivator,而我在解析某个类型时遇到问题。我确信对于有更多仿制药经验的人来说,这将是一个简单的答案。
我传入的容器来自HostContext.Container
using Hangfire;
using System;
using System.Collections.Generic;
using System.Linq;
using ServiceStack;
namespace Common.Hangfire
{
public class FunqJobActivator : JobActivator
{
private Funq.Container _container;
public FunqJobActivator(Funq.Container container)
{
if (container == null)
{
throw new ArgumentNullException("container");
}
_container = container;
}
public override object ActivateJob(Type type)
{
return _container.Resolve<type>(); //** this doesn't compile
}
}
}
答案 0 :(得分:3)
虽然Funq是带有通用API的类型化IOC,但您可以添加帮助程序扩展方法以使用运行时类型解析实例,例如:
public static class ContainerExtensions
{
public static object TryResolve(this Container container, Type type)
{
var mi = typeof(Container).GetMethods(BindingFlags.Public | BindingFlags.Instance)
.First(x => x.Name == "TryResolve" &&
x.GetGenericArguments().Length == 1 &&
x.GetParameters().Length == 0);
var genericMi = mi.MakeGenericMethod(type);
var instance = genericMi.Invoke(container, new object[0]);
return instance;
}
}
这将允许您使用运行时类型解析已注册的依赖项,例如:
var container = new Container();
container.Register(c => new Foo());
var instance = container.TryResolve(typeof(Foo));
答案 1 :(得分:2)
我建议使用不同的IoC框架,因为Funq
不支持采用Type
参数的解析方法,即它没有方法
object Container.Resolve(Type theType);
因此,与Hangfire结婚很困难,因为Hangfire没有提供你可以用作的超载:
virtual T ActivateJob<T>() where T : class
如果你坚持使用Funq,可以(低效率)解决这个问题:
public class FunqJobActivator : JobActivator
{
private const BindingFlags flags = BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance;
private static readonly _activateMethod =
typeof(FunqJobActivator).GetMethod("InternalActivateJob", flags);
private Funq.Container _container;
public FunqJobActivator(Funq.Container container)
{
if (container == null)
{
throw new ArgumentNullException("container");
}
_container = container;
}
public override object ActivateJob(Type type)
{
// this will allow calling InternalActivateJob<T> with typeof(T) == type.
var method = _activateMethod.MakeGenericMethod(new [] { type });
return method.Invoke(this, null);
}
private object InternalActivateJob<T>() where T : class
{
return _container.Resolve<T>();
}
}
答案 2 :(得分:0)
我想您想获得T
并返回T
public T ActivateJob<T>() where T : class
{
return _container.Resolve<T>();
}