如何编写将实现名称与它们实现的接口相关联的自定义属性?

问题描述

我有一个任务,我必须编写一个自定义属性。该属性必须仅适用于接口(最多一次),并允许我将认实现的名称(它必须是一个字符串)与其接口相关联。 (甚至可以有多个实现,但重要的是必须至少有一个认实现)。 以下是我的代码

[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 中的“项目”),指定命名空间和类名就足够了。

顺便说一句,您还可以使用反射来找出哪个类实现了哪个接口 - 没有任何属性。