具有多个自定义属性的PowerShell方法

时间:2016-08-18 19:52:52

标签: c# powershell powershell-v5.0

我无法在PowerShell 5.0类方法上使用多个自定义属性(装饰器)。在C#中,我能够做到以下几点:

public class SomeAttribute : Attribute {
    public string Text { get; set; }
}

public class OtherAttribute : Attribute {
    public string Text { get; set; }
}

public class MyClass {
    [SomeAttribute(Text = "sometext")]
    [OtherAttribute(Text = "othertext")]
    public void MyMethod() {
        // ...
    }
}

然后在其他地方调用

object[] customAttributes = typeof(MyClass).GetMethod("MyMethod").GetCustomAttributes(false);

它给了我一个与该方法相关的所有自定义属性的数组,没有任何问题。

在PowerShell 5.0中,我试图以类似方式使用:

class SomeAttribute : Attribute {
    [string]$Text
}

class OtherAttribute : Attribute {
    [string]$Text
}

class MyClass {
    [SomeAttribute(Text="sometext")]
    [OtherAttribute(Text="othertext")]
    MyMethod() {
        # ...
    }
}

这似乎是正确的语法,因为PowerShell乐意接受它。但是通过

列出属性
[MyClass].GetMethod("MyMethod").GetCustomAttributes($false)

返回以下错误:

Exception calling "GetCustomAttributes" with "1" argument(s): "Could not load type 'OtherAttribute' from assembly '⧹powershell, Version=0.0.0.0, Culture=neutral, PublicKeyToken=null'."
At line:1 char:1
+ [MyClass].GetMethod("MyMethod").GetCustomAttributes($false)
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    + CategoryInfo          : NotSpecified: (:) [], MethodInvocationException
    + FullyQualifiedErrorId : TypeLoadException

[MyClass].GetMethod("MyMethod").CustomAttributes

只返回$null

但是,当我只使用一个自定义属性时,一切都按预期工作,并使用上面的代码段正确返回属性。
如何为PowerShell 5.0类方法正确定义多个自定义属性?

更新 - 仅使用一个自定义属性的行为示例

我们只假设原始问题中的第一个自定义属性。

class SomeAttribute : Attribute {
    [string]$Text
}

class MyClass {
    [SomeAttribute(Text="sometext")]
    MyMethod() {
        # ...
    }
}

然后

[MyClass].GetMethod("MyMethod").GetCustomAttributes($false)

给出以下输出

Text     TypeId
----     ------
sometext SomeAttribute

[MyClass].GetMethod("MyMethod").CustomAttributes

给出以下输出

AttributeType Constructor  ConstructorArguments NamedArguments
------------- -----------  -------------------- --------------
SomeAttribute Void .ctor() {}                   {Text = "sometext"}

这让我相信一个属性确实按预期工作。

2 个答案:

答案 0 :(得分:2)

似乎PowerShell(.NET,恕我直言)在引用具有相同全名的两个不同程序集时遇到了麻烦(有些事情告诉我,这在.NET中是不可能的。)

PS> class A {}
PS> class B {}
PS> [A].Assembly -eq [B].Assembly
False
PS> [A].Assembly.FullName -eq [B].Assembly.FullName
True
PS> class CA { [A] $A }; class CB { [B] $B }
PS> [CA]::new().A #work fine
PS> [CB]::new().B #error

解决方案是在同一个命令中定义类AB,这会将它们放在同一个生成的程序集中:

PS> class A {}; class B {}
PS> [A].Assembly -eq [B].Assembly
True
PS> class CA { [A] $A }; class CB { [B] $B }
PS> [CA]::new().A #work fine
PS> [CB]::new().B #work fine

所以,这对我来说很好:

PS> class SomeAttribute : Attribute {
>>>     [string]$Text
>>> }
>>> class OtherAttribute : Attribute {
>>>     [string]$Text
>>> }
PS> class MyClass {
>>>     [SomeAttribute(Text="sometext")]
>>>     [OtherAttribute(Text="othertext")]
>>>     MyMethod() {
>>>         # ...
>>>     }
>>> }
PS> [MyClass].GetMethod("MyMethod").GetCustomAttributes($false)

Text      TypeId
----      ------
sometext  SomeAttribute
othertext OtherAttribute

其他解决方案是不在交互式会话中定义类,而是在脚本文件中定义类。在这种情况下,模糊文件名将成为生成的程序集名称的一部分,因此即使属性在不同的脚本文件中定义,您也不会遇到此问题。

答案 1 :(得分:0)

我怀疑即使只使用一个自定义属性,您确实得到了预期的结果。原因是方法不支持属性,至少我发现了。方法支持可选的返回类型,在PowerShell中,看起来就像属性。那就是你可以定义一个返回void的方法......

MyMethod() {
    $foo = 5        
}

或返回某些东西的方法(在这种情况下像int一样)......

[int] MyMethod() {
    $foo = 5
    return $foo
}

(我希望有人可以证明我错了,并表明PowerShell类支持属性!)