28

在控制台应用程序中,我正在使用 Log4Net,在 Main 方法中我正在获取记录器对象。现在,我想通过让所有类从具有 ILog 属性并且应该由属性注入而不是构造函数注入设置的 BaseClass 继承来使这个日志对象在我的所有类中可用。

我正在使用 AutoFac IoC 容器,如何将我的日志对象注入到每个类的 Log 属性中?

实现这一目标的最佳/最简单方法是什么?

有没有办法自动解析类型?

下面是我的测试应用程序:

namespace ConsoleApplication1
{
    class Program
    {
        static ILog Log;
        static IContainer Container;

        static void Main(string[] args)
        {                
           InitializeLogger();

           InitializeAutoFac();

            // the below works but could it be done automatically (without specifying the name of each class)?
           Product.Log = Container.Resolve<ILog>();

           // tried below but didn't inject ILog object into the Product
           Container.Resolve<Product>();

           RunTest();

            Console.ReadLine();
        }

        private static void RunTest()
        {
            var product = new Product();
            product.Do();
        }

        private static void InitializeAutoFac()
        {
            var builder = new ContainerBuilder();

            builder.Register(c => Log).As<ILog>();

            builder.RegisterType<Product>().PropertiesAutowired();

            Container = builder.Build();            
        }

        private static void InitializeLogger()
        {
            log4net.Config.XmlConfigurator.Configure();

            Log = LogManager.GetLogger("LoggerName");
        }
    }

    public class Product
    {
        public static ILog Log { get; set; }

        public void Do()
        {
            // this throws exception because Log is not set   
            Log.Debug("some Debug");  
        }
    }
}
4

6 回答 6

26

在我看来,Ninject创建的解决方案比Autofac 中的属性注入要好得多。因此,我创建了一个自定义属性,它是一个 postsharp 方面,它自动注入我的类:

[AutofacResolve]
public IStorageManager StorageManager { get; set; }

我的方面:

[Serializable]
[AttributeUsage(AttributeTargets.Property, AllowMultiple = false)]
public class AutofacResolveAttribute : LocationInterceptionAspect
{
    public override void OnGetValue(LocationInterceptionArgs args)
    {
        args.ProceedGetValue();

        if (!args.Location.LocationType.IsInterface) return;

        if ( args.Value != null )
        {
           args.Value = DependencyResolver.Current.GetService(args.Location.LocationType);
           args.ProceedSetValue();
        }
    }
}

我知道这个问题的答案已经给出,但我认为这是解决 Autofac 中自动属性注入的一种非常巧妙的方法。也许它将来对某人有用。

于 2014-05-08T07:18:09.393 回答
26

使用属性注入

builder.Register(c => LogManager.GetLogger("LoggerName"))
       .As<ILog>();

builder.RegisterType<CustomClass>()
       .PropertiesAutowired();
于 2013-03-24T15:43:24.453 回答
15

属性注入适用于Properties而不是Fields。在您的课程中,Log 是一个字段而不是一个属性,因此它永远不会被 Autofac 解析。

于 2013-11-08T04:48:58.620 回答
3

使用属性注入 (除了@cuongle 答案)。

选项1:

builder.Register(c => LogManager.GetLogger("LoggerName")).As<ILog>();

builder.RegisterType<Product>()
        .WithProperty("Log", LogManager.GetLogger("LoggerName"));

选项 2:

或者你可以SetLog在类中添加一个方法Product

public class Product
{
    public static ILog Log { get; set; }
    public SetLog(Log log)
    {
        this.Log = log;
    }
}

这样您就不必调用LogManager.GetLogger("LoggerName")两次,而是使用构建器的上下文来解析Log.

builder.Register(c => LogManager.GetLogger("LoggerName")).As<ILog>();

builder.Register(c => 
    var product = new Product();
    product.SetLog(c.Resolve<Log>());
    return product;
);

选项 3:

使用OnActvated

一旦组件完全构建,就会引发 OnActivated 事件。在这里,您可以执行依赖于完全构建的组件的应用程序级任务 - 这些应该很少见。

builder.RegisterType<Product>()
    .OnActivated((IActivatedEventArgs<Log> e) =>
    {
        var product = e.Context.Resolve<Parent>();
        e.Instance.SetParent(product);
    });

这些选项提供了更多控制,您不必担心@steven 评论:

然而,PropertiesAutowired 的可怕之处在于它会进行隐式属性注入,这意味着将跳过任何无法解析的依赖项。这很容易错过配置错误,并可能导致应用程序在运行时失败

于 2019-01-22T07:15:54.913 回答
3

我不想使用 postsharp,所以我做了一个快速的解决方案,但它不会自动注入。我是 Autofac 的新手,应该可以在此解决方案的基础上进行构建。

[Serializable]
[AttributeUsage(AttributeTargets.Property)]
public class AutofacResolveAttribute : Attribute
{
}

public class AutofactResolver
{
    /// <summary>
    /// Injecting objects into properties marked with "AutofacResolve"
    /// </summary>
    /// <param name="obj">Source object</param>
    public static void InjectProperties(object obj)
    {
        var propertiesToInject = obj.GetType().GetProperties()
             .Where(x => x.CustomAttributes.Any(y => y.AttributeType.Name == nameof(AutofacResolveAttribute))).ToList();

        foreach (var property in propertiesToInject)
        {
            var objectToInject = Autofact.SharedContainer.Resolve(property.PropertyType);
            property.SetValue(obj, objectToInject, null);
        }
    }
}

将其与此调用一起使用:

AutofactResolver.InjectProperties(sourceObject);
于 2016-02-18T14:04:13.553 回答
0

有一个接口IPropertySelector可以实现并通过实现传递.PropertiesAutowired(new MyPropertySelector())。这将允许您实现您想要的任何逻辑。

于 2022-01-09T13:14:53.200 回答