问题描述
我有一个任务,我必须编写一个自定义属性。该属性必须仅适用于接口(最多一次),并允许我将默认实现的名称(它必须是一个字符串)与其接口相关联。 (甚至可以有多个实现,但重要的是必须至少有一个默认实现)。 以下是我的代码:
[AttributeUsage(AttributeTargets.Interface)]
public class ImplementationNameAttribute:Attribute
{
public List<string> Implementations { get; set; }
public ImplementationNameAttribute(List<string> implementations)
{
foreach(var i in implementations)
Implementations.Add(i);
}
}
[ImplementationName(/*what should I write here?*/)]
public interface I
{
}
public class C:I
{
}
如果参数是字符串,我如何使用GetType()来检查某个对象是否确实是一个类并且是接口I的实现?请记住,赋值表示参数是字符串而不是对象。
解决方法
您不能将 List<T>
传递给属性,但可以将字符串数组传递给它。
有了列表,编译器会生成一个
CS0181 属性构造函数参数“实现”的类型为“列表”,这不是有效的属性参数类型
[AttributeUsage(AttributeTargets.Interface)]
public class ImplementationNameAttribute : Attribute
{
public string[] Implementations { get; }
public ImplementationNameAttribute(params string[] implementations)
{
Implementations = implementations;
}
}
[ImplementationName("IA","IB","IC")]
public interface I
{
}
如果没有 params
关键字,您将不得不写
[ImplementationName(new[] { "IA","IC" })]
当然,您仍然可以保留属性的 List<string>
并将其添加到字符串数组参数中的元素。
可以使用 nameof 关键字在编译时测试这些接口是否存在。它只是为您传递的标识符返回一个字符串常量。
标识符必须对应于现有的变量、类型、命名空间或成员。 nameof
始终只返回限定标识符的最后一部分。例如,nameof(N.IA)
返回 "IA"
。如果您需要全名,则必须写 nameof(N) + "." + nameof(IA)
。
[ImplementationName(nameof(IA),nameof(IB),nameof(IC))]
nameof
当然不会测试您是否向它传递了接口名称或其他内容。此外,这种方法仅在编译时已知类型时才有效。如果它在运行时加载的程序集中,事情就会变得棘手。请参阅此对"Resolve Type from Class Name in a Different Assembly"的回答。
Type.GetType("typename",false)
如果类型不存在但不抛出异常,则返回 null
。你可以用它来测试类型是否存在。
如果参数是字符串,我如何使用GetType()来检查某个对象是否确实是一个类并且是接口I的实现?请记住,赋值表示参数是字符串而不是对象。
要从 System.Type
创建 string
的实例,请使用 Type.GetType(stringQualifier)
。
https://docs.microsoft.com/en-us/dotnet/api/system.type.gettype?view=net-5.0#system-type-gettype(system-string)
由字符串限定的类型所在的程序集必须在执行此操作时加载。此外,它取决于 .NET Framework 版本,必须指定您作为字符串传入的类型名的精确程度。
.NET 中的完全限定类型名不仅是类名,还包括命名空间和程序集名称,包括 GUID、语言和公钥标记。
也许如果接口和类属于同一个程序集(VS 中的“项目”),指定命名空间和类名就足够了。
顺便说一句,您还可以使用反射来找出哪个类实现了哪个接口 - 没有任何属性。