有時在撰寫程式為了增加彈性或是便利性,我們可考慮使用Attribute來為類別、方法、欄位、與屬性附加一些資訊。這樣的設計方式在.NET中已經存在許久,這類應用也越來越普及,相信大家就算沒有用過也有看過,像是製作控制項時會用到的Browsable、Description、Category…,做序列化時會用到的XmlIgnore、XmlElement…,許多地方都會用到這樣的小技巧。

在Attribute的建立上,簡單的說只是一個繼承Attribute的類別,裡面存放著所要附加的資料,並透過AttributeUsage屬性設定該Attribute所能使用的範圍。像是:

<AttributeUsage(AttributeTargets.Property Or AttributeTargets.Field)>
Class NewPropertyOrFieldAttribute
    Inherits Attribute

    Property Data as Object

End Class

這邊以利用Attribute將資訊附加在列舉成員欄位為例,假設我們有要做類似於Office這樣的軟體需求,為求方便使用,可能會希望開出一個建構子或是成員方法,裡面可帶入列舉,依帶入的列舉值開啟對應的應用程式。這時就可以透過Attribute為列舉的成員欄位附加對應要開啟的類別資訊,像是下面這樣:

NotInheritable Class RelativedTypeAttribute
    Inherits Attribute

    Property RelativedType As Type

    Sub New(ByVal relativedType As Type)
        Me.RelativedType = relativedType
    End Sub

End Class

Enum ApplicationType
    <RelativedType(GetType(LevelUp.Office.Word))> _
    Word

    <RelativedType(GetType(LevelUp.Office.Excel))> _
    Excel

    <RelativedType(GetType(LevelUp.Office.Access))> _
    Access

    <RelativedType(GetType(LevelUp.Office.PowerPoint))> _
    PowerPoint

    <RelativedType(GetType(LevelUp.Office.Visio))> _
    Visio
End Enum

再透過反射取得列舉成員欄位對應的類別資訊,將其類別動態建立叫起使用就可以了。

Sub Main()
    StartApplication(ApplicationType.Word)
End Sub

Sub StartApplication(ByVal apType As ApplicationType)
    Dim type As Type = apType.GetType()
    Dim field = type.GetField(apType.ToString)
    Dim att = field.GetCustomAttributes(GetType(RelativedTypeAttribute), False).Cast(Of RelativedTypeAttribute)().FirstOrDefault()
    Activator.CreateInstance(att.RelativedType).Start()
End Sub

又或是我們有需求透過列舉取得對應的描述字串,我們可以使用Attribute為列舉的成員欄位附加對應的資源編號,像是下面這樣:

NotInheritable Class ResouceIDAttribute
    Inherits Attribute

    Property ResourceID As String

    Sub New(ByVal resourceID As String)
        Me.ResourceID = resourceID
    End Sub

End Class

Enum InterfaceType
    <ResouceID("RS232_InterfaceName")> _
    RS232

    <ResouceID("RS485_InterfaceName")> _
    RS485

    <ResouceID("GPIB_InterfaceName")> _
    GPIB

    <ResouceID("I2C_InterfaceName")> _
    I2C
End Enum

如此可透過反射去取得附加的資源ID,進一步取得資源檔所設定的字串。

Sub Main()
    Dim instrumentInterface As InterfaceType
    Dim type As Type = instrumentInterface.GetType()
    Dim field = type.GetField(instrumentInterface.ToString)
    Dim resourceID = field.GetCustomAttributes(GetType(ResouceIDAttribute), False).Cast(Of ResouceIDAttribute)().FirstOrDefault.ResourceID
    Console.WriteLine(My.Resources.ResourceManager.GetString(resourceID))
End Sub

當然還可以用在很多地方,這邊只是舉幾個例子稍作紀錄。