2

我正在尝试使用ServiceStack IoC 为HangFire编写 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
        }
    }
}
4

3 回答 3

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));
于 2015-04-21T14:58:07.113 回答
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>();
    }
}
于 2015-04-21T14:24:20.970 回答
0

我想你想得到一个T并返回一个T

public T ActivateJob<T>() where T : class
{
    return _container.Resolve<T>();  
}
于 2015-04-21T14:03:52.263 回答