- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
这将有些疯狂,但是我相信,如果可能的话,它将成为当前任务中最可维护的解决方案。
我们的应用程序使用Autofac进行依赖项注入。
我们使用自定义数据文件格式,出于技术(性能/存储空间优化)或域方面的原因,我们需要能够对其进行发展。该应用程序将始终只写入该格式的最新版本,但也需要能够读取所有以前的版本。通常,版本之间的演变是相当渐进的,仅在几个地方进行了更改,因此,许多用于阅读它的代码将保持不变。
文件格式版本号在文件开头存储为整数。读取任何版本的文件格式将始终导致相同的数据结构,在此称为Scenario
。
可以从文件读取数据的类依赖于IReadDataFile
:
public interface IReadDataFile
{
Scenario From(string fileName);
}
ReadDataFileContents : IReadDataFileContents
└> ReadCoreData : IReadCoreData
└> ReadAdditionalData : IReadAdditionalData
└> NormalizeName : INormalizeName
ReadDataFileContentsV2 : IReadDataFileContents
└> ReadCoreData : IReadCoreData
└> ReadAdditionalDataV2 : IReadAdditionalData
└> NormalizeNameV2 : INormalizeName
└> AdditionalNameRegex : IAdditionalNameRegex
ReadDataFileContentsV2 : IReadDataFileContents
└> ReadCoreData : IReadCoreData
└> ReadAdditionalDataV3 : IReadAdditionalData
└> NormalizeNameV2 : INormalizeName
└> AdditionalNameRegexV3 : IAdditionalNameRegex
IReadDataFile.From()
方法加载文件时,它都需要获取文件格式版本的适当子图。一个简单的方法是通过注入工厂:
public class ReadDataFile : IReadDataFile
{
private readonly IGetDataFileVersion getDataFileVersion;
private readonly Func<int, IReadDataFileContents> createReadDataFileContents;
public ReadDataFile(
IGetDataFileVersion getDataFileVersion,
Func<int, IReadDataFileContents> createReadDataFileContents)
{
this.getDataFileVersion = getDataFileVersion;
this.createReadDataFileContents = createReadDataFileContents;
}
public Scenario From(string fileName)
{
var version = this.getDataFileVersion.From(fileName);
var readDataFileContents = this.createReadDataFileContents(version);
return readDataFileContents.From(fileName);
}
}
Keyed<T>
非常麻烦,并且容易出错,并且无法很好地扩展其他文件格式版本(尤其是图比示例复杂得多)。
builder.RegisterAssemblyTypes(typeof(IReadDataFile).Assembly).AsImplementedInterfaces();
builder.RegisterType<ReadDataFileContents>().As<IReadDataFileContents>();
builder.RegisterType<ReadDataFileContentsV2>().Keyed<IReadDataFileContents>(2);
builder.RegisterType<ReadAdditionalData>().As<IReadAdditionalData>();
builder.RegisterType<ReadAdditionalDataV2>().Keyed<IReadAdditionalData>(2);
builder.RegisterType<ReadAdditionalDataV3>().Keyed<IReadAdditionalData>(3);
builder.RegisterType<NormalizeName>().As<INormalizeName>();
builder.RegisterType<NormalizeNameV2>().Keyed<INormalizeName>(2);
builder.RegisterType<AdditionalNameRegex>().As<IAdditionalNameRegex>();
builder.RegisterType<AdditionalNameRegexV3>().Keyed<IAdditionalNameRegex>(3);
builder.Register<Func<int, IReadDataFileContents>>(c =>
{
var context = c.Resolve<IComponentContext>();
return version => // magic happens here
});
2
的解析失败时,将解析默认注册。
createReadDataFileContents
值
version
调用
3
工厂,因此所需的图形是上述文件格式版本3的图形。
IReadDataFileContents
解析
3
。这是不成功的。没有这样的注册。
IReadDataFileContents
解析
2
。这样成功了。
IReadCoreData
。尝试使用键
3
,然后按
2
来解决此问题。都失败,因此默认注册得以解决,从而成功。
IReadAdditionalData
;尝试使用键
3
解决此问题,该键成功。
INormalizeName
;键
3
的解析失败,然后
2
的尝试成功。
IAdditionalNameRegex
;键
3
的解析尝试成功。
version
的初始值开始解决的每个独立依赖项,都需要进行版本“倒数”回退过程。
Module.AttachToComponentRegistration()
-我已经在其他地方使用它来使用
registration.Preparing
参与解析过程;但是,只有在找到合适的注册后才引发该事件,并且在此之前似乎没有引发过的事件,也没有在解决方法失败的情况下注册回调的方法(这使我感到惊讶)。
IRegistrationSource
-这似乎是实现更通用的注册/解决方案原则的方法,但是如果我确实正在寻找这个地方,我无法理解其中需要做的事情对于。
WithKeyAttribute
-我们不能在这里使用它,因为我们需要控制从外部注入的依赖项的“版本”(而且,实际的业务代码将变得依赖于Autofac,这永远都不好。)
ILifetimeScope.ResolveOperationBeginning
-这看起来非常有希望,但是该事件仅针对已经成功的解决方案引发。
IIndex<TKey, TValue>
-乍看起来确实不错的另一件事,但是它包含已经构造的实例,因此无法获得较低级别分辨率的版本密钥。
最佳答案
开箱即用,Autofac确实没有这种级别的控制。但是,如果您不介意一点间接性,可以在中间添加一个工厂来构建它。
首先,让我发布一个有效的C#文档,然后对其进行解释。您应该可以将其粘贴到例如.csx
scriptcs文档中,然后查看它的内容-这就是我编写的地方。
using Autofac;
using System.Linq;
// Simple interface just used to prove out the
// dependency chain that gets resolved.
public interface IDependencyChain
{
IEnumerable<Type> DependencyChain { get; }
}
// File reading interfaces
public interface IReadDataFileContents : IDependencyChain { }
public interface IReadCoreData : IDependencyChain { }
public interface IReadAdditionalData : IDependencyChain { }
public interface INormalizeName : IDependencyChain { }
public interface IAdditionalNameRegex : IDependencyChain { }
// File reading implementations
public class ReadDataFileContents : IReadDataFileContents
{
private readonly IReadCoreData _coreReader;
private readonly IReadAdditionalData _additionalReader;
public ReadDataFileContents(IReadCoreData coreReader, IReadAdditionalData additionalReader)
{
this._coreReader = coreReader;
this._additionalReader = additionalReader;
}
public IEnumerable<Type> DependencyChain
{
get
{
yield return this.GetType();
foreach(var t in this._coreReader.DependencyChain)
{
yield return t;
}
foreach(var t in this._additionalReader.DependencyChain)
{
yield return t;
}
}
}
}
public class ReadDataFileContentsV2 : ReadDataFileContents
{
public ReadDataFileContentsV2(IReadCoreData coreReader, IReadAdditionalData additionalReader)
: base(coreReader, additionalReader)
{
}
}
public class ReadCoreData : IReadCoreData
{
public IEnumerable<Type> DependencyChain
{
get
{
yield return this.GetType();
}
}
}
public class ReadAdditionalData : IReadAdditionalData
{
private readonly INormalizeName _normalizer;
public ReadAdditionalData(INormalizeName normalizer)
{
this._normalizer = normalizer;
}
public IEnumerable<Type> DependencyChain
{
get
{
yield return this.GetType();
foreach(var t in this._normalizer.DependencyChain)
{
yield return t;
}
}
}
}
public class ReadAdditionalDataV2 : ReadAdditionalData
{
public ReadAdditionalDataV2(INormalizeName normalizer)
: base(normalizer)
{
}
}
public class ReadAdditionalDataV3 : ReadAdditionalDataV2
{
public ReadAdditionalDataV3(INormalizeName normalizer)
: base(normalizer)
{
}
}
public class NormalizeName : INormalizeName
{
public IEnumerable<Type> DependencyChain
{
get
{
yield return this.GetType();
}
}
}
public class NormalizeNameV2 : INormalizeName
{
public readonly IAdditionalNameRegex _nameRegex;
public NormalizeNameV2(IAdditionalNameRegex nameRegex)
{
this._nameRegex = nameRegex;
}
public IEnumerable<Type> DependencyChain
{
get
{
yield return this.GetType();
foreach(var t in this._nameRegex.DependencyChain)
{
yield return t;
}
}
}
}
public class AdditionalNameRegex : IAdditionalNameRegex
{
public IEnumerable<Type> DependencyChain
{
get
{
yield return this.GetType();
}
}
}
public class AdditionalNameRegexV3 : AdditionalNameRegex { }
// File definition modules - each one registers just the overrides needed
// for the upgraded version of the file type. ModuleV1 registers the base
// stuff that will be used if things aren't overridden. If any version
// of a file format needs to "revert back" to an old mechanism, like if
// V2 needs NormalizeNameV2 and V3 needs NormalizeName, you'd have to re-register
// the base NormalizeName in the V3 module - override the override.
public class ModuleV1 : Module
{
protected override void Load(ContainerBuilder builder)
{
builder.RegisterType<ReadDataFileContents>().As<IReadDataFileContents>();
builder.RegisterType<ReadCoreData>().As<IReadCoreData>();
builder.RegisterType<ReadAdditionalData>().As<IReadAdditionalData>();
builder.RegisterType<NormalizeName>().As<INormalizeName>();
}
}
public class ModuleV2 : Module
{
protected override void Load(ContainerBuilder builder)
{
builder.RegisterType<ReadDataFileContentsV2>().As<IReadDataFileContents>();
builder.RegisterType<ReadAdditionalDataV2>().As<IReadAdditionalData>();
builder.RegisterType<NormalizeNameV2>().As<INormalizeName>();
builder.RegisterType<AdditionalNameRegex>().As<IAdditionalNameRegex>();
}
}
public class ModuleV3 : Module
{
protected override void Load(ContainerBuilder builder)
{
builder.RegisterType<ReadAdditionalDataV3>().As<IReadAdditionalData>();
builder.RegisterType<AdditionalNameRegexV3>().As<IAdditionalNameRegex>();
}
}
// Something has to know about how file formats are put together - a
// factory of some sort. Here's the thing that "knows." You could probably
// drive this from config or something else, too, but the idea holds.
public class FileReaderFactory
{
private readonly ILifetimeScope _scope;
public FileReaderFactory(ILifetimeScope scope)
{
// You can always resolve the current lifetime scope as a parameter.
this._scope = scope;
}
public IReadDataFileContents CreateReader(int version)
{
using(var readerScope = this._scope.BeginLifetimeScope(b => RegisterFileFormat(b, version)))
{
return readerScope.Resolve<IReadDataFileContents>();
}
}
private static void RegisterFileFormat(ContainerBuilder builder, int version)
{
switch(version)
{
case 1:
builder.RegisterModule<ModuleV1>();
break;
case 2:
builder.RegisterModule<ModuleV1>();
builder.RegisterModule<ModuleV2>();
break;
case 3:
default:
builder.RegisterModule<ModuleV1>();
builder.RegisterModule<ModuleV2>();
builder.RegisterModule<ModuleV3>();
break;
}
}
}
// Only register the factory and other common dependencies - not the file
// format readers. The factory will be responsible for managing the readers.
// Note that since readers do resolve from a child of the current lifetime
// scope, they can use common dependencies that you'd register in the
// container.
var builder = new ContainerBuilder();
builder.RegisterType<FileReaderFactory>();
var container = builder.Build();
using(var scope = container.BeginLifetimeScope())
{
var factory = scope.Resolve<FileReaderFactory>();
for(int i = 1; i <=3; i++)
{
Console.WriteLine("Version {0}:", i);
var reader = factory.CreateReader(i);
foreach(var t in reader.DependencyChain)
{
Console.WriteLine("* {0}", t);
}
}
}
Version 1:
* Submission#0+ReadDataFileContents
* Submission#0+ReadCoreData
* Submission#0+ReadAdditionalData
* Submission#0+NormalizeName
Version 2:
* Submission#0+ReadDataFileContentsV2
* Submission#0+ReadCoreData
* Submission#0+ReadAdditionalDataV2
* Submission#0+NormalizeNameV2
* Submission#0+AdditionalNameRegex
Version 3:
* Submission#0+ReadDataFileContentsV2
* Submission#0+ReadCoreData
* Submission#0+ReadAdditionalDataV3
* Submission#0+NormalizeNameV2
* Submission#0+AdditionalNameRegexV3
IDisposable
依赖项。如果您采用这种方式,并且任何文件读取依赖项都是可抛弃的,则需要将其注册为
Owned
或其他名称,这样工厂内部的微小子生存期范围不会实例化,然后立即处理您要处理的东西会需要的。
InstancePerOwned
东西的工作方式。它在幕后有先例。
Func<int, IReadDataFileContents>
方法,则可以让它解析工厂并在其中调用
CreateReader
方法。
关于c# - 使用级联后备解决键控子图,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32897868/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!