使用泛型隐式运算符有什么问题
本文关键字:什么 问题 运算符 泛型 | 更新日期: 2023-09-27 18:20:36
如果我在非泛型生成器类中使用隐式运算符,一切都可以:
public class ReligionBuilder
{
private Religion _religion;
public ReligionBuilder()
{
_religion = new Religion(){//some codes}
}
public ReligionBuilder AddToRepository()
{
Repository<Religion>.Add(_religion);
return this;
}
public Religion Build()
{
return _religion;
}
public static implicit operator Religion(ReligionBuilder _builder)
{
return _builder.Build();
}
}
我可以使用它:
Religion religion=new ReligionBuilder().AddToRepository();
但是,如果这个运算符在泛型类中,则会出现错误:
public abstract class DataTestBuilderBase<T> : IDataTestBuilder<T>
{
protected T TestData { get; set; }
public virtual T Build()
{
return TestData;
}
public abstract IDataTestBuilder<T> AddToRepository();
public abstract IDataTestBuilder<T> WithDefault();
public static implicit operator T(DataTestBuilderBase<T> builder)
{
return builder.Build();
}
}
public class PersonDataTestBuilder : DataTestBuilderBase<Person>
{
private Person _person;
public PersonDataTestBuilder()
{
//some codes
}
public override IDataTestBuilder<Person> AddToRepository()
{
//some codes
return this;
}
}
用法:
PersonDataTestBuilder _testBuilder = new PersonDataTestBuilder();
Person person = _testBuilder.AddToRepository();
错误是:无法将IDataTestBuilder转换为Person
问题出在哪里?
AddToRepository
只返回IDataTestBuilder<Person>
(根据编译时返回类型),并且没有从到Person
的隐式转换。如果您将抽象方法的返回类型更改为DataTestBuilderBase<T>
,那么它应该可以工作——尽管坦率地说,我无论如何都不想使用隐式转换。我通常对提供隐式转换非常谨慎——它们通常会让代码变得不那么清晰,就像我相信它们在这里所做的那样。
无论如何,您真的应该在构建器上提供AddToRepository
吗?对于一个建筑商来说,这感觉是一个不合适的行为——我想:
Person person = new PersonBuilder { /* properties */ }
.Build()
.AddToRepository();
编辑:为了说明我对更改AddToRepository
的返回类型的意思,这里有一个简短但完整的程序来演示。它运行良好。
using System;
public abstract class BuilderBase<T>
{
public abstract T Build();
public abstract BuilderBase<T> AddToRepository();
public static implicit operator T(BuilderBase<T> builder)
{
return builder.Build();
}
}
public class TestBuilder : BuilderBase<string>
{
public override string Build()
{
return "Built by Build()";
}
public override BuilderBase<string> AddToRepository()
{
return this;
}
}
class Program
{
static void Main(string[] args)
{
string x = new TestBuilder().AddToRepository();
Console.WriteLine(x);
}
}