c# - 使用带有目标对象信息的 MEF 注入(inject)对象

标签 c# logging dependency-injection mef

我正在寻找一种使用 MEF 将 log4net 记录器包装到各种对象中的日志对象注入(inject)方法。我目前遇到的问题是日志记录对象需要它所属的对象的类型。我可以通过在包含对象的构造函数中设置日志记录对象的类型属性来解决这个问题,但是这给开发人员留下了设置类型的责任,并且我可以想到没有编译时间限制来强制执行此操作.

有没有办法指定当日志对象由MEF生成并注入(inject)时,其构造函数参数设置为注入(inject)的目标类的类型?

我的记录器实现了一个接口(interface)

public interface ILogger
{
    Type Type { get; }
}

具体实现的一个例子是

[Export(typeof(Ilogger))]
public class SimpleLogger : ILogger
{
    public SimpleLogger(Type typeOfObjectToLogFor)
    {
         this.Type = typeOfObjectToLogFor
    }

    public Type Type { get; }

    public void Info(string message)
    {
        //log the messsage including the type information
    }
}

并且它目前不使用 MEF 作为:

public class ExampleObject
{
    private readonly ILogger logger = new SimpleLogger(typeof(ExampleObject));

    public ExampleObject(){}

    public void MethodThatLogs()
    {
        logger.Info("MethodThatLogs called");
    }
}

我想做的是使用构造函数注入(inject)来注入(inject)它:

public class ExampleObject
{
    private readonly ILogger logger;

    [ImportingConstructor]
    public ExampleObject(Ilogger injectedLogger)
    {
        logger = injectedLogger;
    }

    public void MethodThatLogs()
    {
        logger?.Info("MethodThatLogs called");
    }
}

我可以用惰性评估反射来完成所有这些,但感觉应该可以从一个体面的 DI 容器中实现,希望这意味着 MEF 会支持它,有人帮忙吗?

最佳答案

默认情况下,指定 [Export] 属性,您将 PartCreationPolicy 设置为 Shared,这意味着容器将创建用于导出的单例 - 您的记录器。

但我建议您不要导出一个类,而是导出一个工厂方法,它将接受一个参数并为您创建记录器。

class LoggerFactory
{
    [Export("GetLogger")]
    public ILogger GetLogger(Type type)
    {
        return new SimpleLogger(type);
    }
}

class ExampleObject
{
    private readonly ILogger logger;

    [ImportingConstructor]
    public ExampleObject([Import(ContractName = "GetLogger", AllowDefault = true)]Func<Type, ILogger> loggerCreator)
    {
        logger = loggerCreator?.Invoke(this.GetType());
    }
}

关于c# - 使用带有目标对象信息的 MEF 注入(inject)对象,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35840106/

相关文章:

java - Log4j2 用零填充文件名

python - 为什么控制台记录器与打印不同步?

batch-file - 批量设置=log.txt

scala - 如何避免Scala中的依赖注入(inject)?

azure - .Net6 Azure Functions 中的 ILogger 依赖项注入(inject)问题

c# - 在 VB.net 中比较可为 null 的 DateTime

c# - 如何将多个操作数应用于单个运算符?

c# - 常量字符串声明/初始化断点

c# - 我的 Azure Function App 在哪个端口上运行?

dependency-injection - 缺少方法异常 : No parameterless constructor defined for type