如何创建匿名类,以便将其传递给泛型构造函数
本文关键字:构造函数 泛型 何创建 创建 | 更新日期: 2023-09-27 18:09:36
我正在将非通用存储库的测试移植到通用存储库。而不是使用Repo<MyType>
,我想用Repo<object>
或其他东西来测试更抽象的方法。
我尝试使用一个匿名的实例,但是它不工作:
[Test]
public void AddToRepoIncrementsByOne()
{
IFile file = new FileFake();
IByteSerializer<object> serializer = new SerializerObjectFake();
repo = new WriteableFileRepository<object>(file, serializer);
var obj = new { Name = "test" };
repo.Add(obj);
Assert.True(repo.Items.Count() == 1);
Assert.AreEqual(repo.Items.Single().Name, obj.Nome);
//----------------------------------^^^^
//-------------------compiler error here
}
我试图从对象中推断类引用,但也没有成功:
var obj = new { Nome = "teste" };
Type T = obj.GetType();
IArquivo arquivo = new ArquivoFake();
IByteSerializer<T> serializer = new SerializerObjectFake();
//--------------^
repo = new RepositórioArquivoEscrita<T>(arquivo, serializer);
我该如何解决这个问题?我不要求使用匿名对象,我只是认为这将是一个实用的方式,使我的测试运行,但我肯定可以做其他方式,如果这不是一个好的设计。
匿名类型的技巧是按例强制转换;让编译器利用类型推断为您完成工作:
public static T CastByExample(this object obj, T prototype) =>
(T)obj;
现在你需要做的就是:
Assert.AreEqual(repo.Items.Single().CastByExample(obj).Name, obj.Name);
而不是:
IByteSerializer<object> serializer = new SerializerObjectFake();
repo = new WriteableFileRepository<object>(file, serializer);
创建一个通用方法来创建您的存储库:
RepoType<T> CreateRepository<T>(IFile file, T objectForType)
{
IByteSerializer<T> serializer = new SerializerObjectFake();
return new WriteableFileRepository<T>(file, serializer);
}
用法:
[Test]
public void AddToRepoIncrementsByOne()
{
IFile file = new FileFake();
var obj = new { Name = "test" };
/*Your type is missing*/ repo = CreateRepository(file, obj);
repo.Add(obj);
Assert.True(repo.Items.Count() == 1);
Assert.AreEqual(repo.Items.Single().Name, obj.Nome);
//----------------------------------^^^^
//-------------------no more error here
}
通过传递匿名对象作为参数(即使你不使用它),它允许你获得它的类型作为泛型参数。然后,您可以使用它来实例化对象,这取决于它的类型。
修复这个最简单的方法是通过使用obj
的通用方法让编译器填补空白:
var obj = new { Nome = "teste" };
Foo(obj);
...
void Foo<T>(T obj) {
IArquivo arquivo = new ArquivoFake();
IByteSerializer<T> serializer = new SerializerObjectFake();
// etc
}
现在T
是匿名类型