PerRequestLifetimeManager can only be used in the context of an HTTP request

后端 未结 2 1385
既然无缘
既然无缘 2021-01-17 21:31

I have a MVC application that uses Unity as its IoC container and have multiple services defined in my application using the PerRequestLifetimeManager.

相关标签:
2条回答
  • 2021-01-17 22:07

    You are using Serivice Location which is considered an anti-pattern.

    Having said that, here is a direct answer to your question:

    One way to solve your problem is using named registrations:

    Let say that you are registering IService to Service using the PerRequestLifetimeManager lifetime manager like this:

    container.RegisterType<IService, Service>(new PerRequestLifetimeManager());
    

    You can also add another registration for the same types but with a different lifetime manager. However, to distinguished between this and the previous registration, you have to give it a name like this:

    container.RegisterType<IService, Service>("transient_service", new TransientLifetimeManager());
    

    Here I am registering IService with Service and using the transient lifetime manager. The name I am giving to this registration is "transient_service" , but you can use any name here.

    Now, from your background thread, you can locate this service like this:

    var service = container.Resolve<IService>("transient_service");
    

    I am assuming here that you have access to the container (which you are doing through the service locator). You might need to update your service locator to enable it to locate services by name.

    UPDATE:

    Here is another solution:

    You can create a custom lifetime manager that acts as the PerRequestLifetimeManager lifetime manager if there is an HttpContext in the current thread, and that will fallback to a TransientLifetimeManager if there isn't.

    Here is how such lifetime manager would look like:

    public class PerRequestOrTransientLifeTimeManager : LifetimeManager
    {
        private readonly PerRequestLifetimeManager m_PerRequestLifetimeManager = new PerRequestLifetimeManager();
        private readonly TransientLifetimeManager m_TransientLifetimeManager = new TransientLifetimeManager();
    
        private LifetimeManager GetAppropriateLifetimeManager()
        {
            if (System.Web.HttpContext.Current == null)
                return m_TransientLifetimeManager;
    
            return m_PerRequestLifetimeManager;
        }
    
        public override object GetValue()
        {
            return GetAppropriateLifetimeManager().GetValue();
        }
    
        public override void SetValue(object newValue)
        {
            GetAppropriateLifetimeManager().SetValue(newValue);
        }
    
        public override void RemoveValue()
        {
            GetAppropriateLifetimeManager().RemoveValue();
        }
    }
    

    You need to modify your registrations to use such lifetime manager.

    UPDATE 2:

    The custom LifetimeManger code won't work with Unity 3.0 or later since it was completely rewritten and further abstracted into new Nuget packages as well. Here is an updated code:

    public class PerRequestOrTransientLifeTimeManager : LifetimeManager
    {
        private readonly PerRequestLifetimeManager _perRequestLifetimeManager = new PerRequestLifetimeManager();
        private readonly TransientLifetimeManager _transientLifetimeManager = new TransientLifetimeManager();
    
        private LifetimeManager GetAppropriateLifetimeManager()
        {
            if (HttpContext.Current == null)
            {
                return _transientLifetimeManager;
            }
    
            return _perRequestLifetimeManager;
        }
    
        public override object GetValue(ILifetimeContainer container = null)
        {
            return GetAppropriateLifetimeManager().GetValue();
        }
    
        public override void SetValue(object newValue, ILifetimeContainer container = null)
        {
            GetAppropriateLifetimeManager().SetValue(newValue);
        }
    
        public override void RemoveValue(ILifetimeContainer container = null)
        {
            GetAppropriateLifetimeManager().RemoveValue();
        }
    
        protected override LifetimeManager OnCreateLifetimeManager()
        {
            return this;
        }
    }
    
    0 讨论(0)
  • 2021-01-17 22:08

    I would suggest you to have 2 separate containers, with different configuration, for the web environment and for the background environment. So, for your web environment, you can control the lifetime per request and in a background task you can do it per thread.

    As you are using service locator, you could have 2 locators, like WebServiceLocator.Resolve<> and BackgroundServiceLocator.Resolve<>

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