如何创建匿名类,以便将其传递给泛型构造函数

本文关键字:构造函数 泛型 何创建 创建 | 更新日期: 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是匿名类型