如何在Powershell中使用扩展方法?

时间:2014-09-18 14:28:31

标签: c# .net powershell extension-methods

我有以下代码:

using System
public static class IntEx
{
    /// <summary>
    /// Yields a power of the given number
    /// </summary>
    /// <param name="number">The base number</param>
    /// <param name="powerOf">the power to be applied on te base number</param>
    /// <returns>Powers applied to  the base number</returns>
    public static IEnumerable<int> ListPowersOf(this int number, int powerOf)
    {
        for (var i = number; ; i <<= powerOf)
        {
            yield return i;
        }
    }
}

我已经在Powershell中加载了dll(Windows 8)。我尝试以下列方式使用它:

$test = 1.ListPowersOf(2)

应该返回@(1,2,4,8,16 ......)

相反,它说没有这样的方法。

我尝试了以下内容:

[BaseDllNamespace]::ListPowersOf(1,2)

仍然没有。我在IntEx类中没有命名空间。

如何让它发挥作用

3 个答案:

答案 0 :(得分:12)

试试这个:

[IntEx]::ListPowersOf(1,2)

[IntEx] | gm -Static -Type Method

列出可用的静态方法。

您还可以使用反射来获取导出类型的列表,以查看您的类型是否可用:

[Reflection.Assembly]::LoadFile('C:path\to.dll')|select -ExpandProperty ExportedTypes

答案 1 :(得分:0)

如果您真的希望它更接近点语法,则可以对IntEx类型使用隐式或显式转换运算符,并将其用作POCO而不是静态扩展名。

可能看起来像这样...

$code=@'
public class MyExtClass {

    public string TheStringValue {get; set;}

    public MyExtClass(string theString){
        TheStringValue = theString;
    }

    public int NumberOf(string target)
    {
        return TheStringValue.Length - TheStringValue.Replace(target, "").Length;
    }

    public static implicit operator MyExtClass(string value){
        return new MyExtClass(value);
    }
}
'@
add-type -TypeDefinition $code

$str_eee = "TheTheThe"
$numberOfEs = ([MyExtClass]$str_eee).NumberOf("e")

if(3 -eq $numberOfEs){Write-Host PASS; exit 0}

Write-Host "FAIL";
exit 1

答案 2 :(得分:0)

我意识到这在这一点上不太可能适用于 OP,但这个问题是我搜索的第一个结果,所以我会在这里发布我的想法。

我很惊讶这个还没有被提及,但是 PowerShell 中的 CodeMethods 本质上是为给定类型编译的扩展方法。它们也很容易编写 - 特别是当您已经有了扩展方法时。

public static class MyStringExtensions
{
    public static string Append(this string source, params char[] characters)
    {
        foreach (var c in characters)
        {
            source += c;
        }
        return source;
    }
    // named PSAppend instead of Append. This is just a naming convention I like to use,
    // but it seems some difference in name is necessary if you're adding the type data 
    // via a types.ps1xml file instead of through the Update-TypeData command
    public static string PSAppend(PSObject source, params char[] characters)
    {
        if (source.BaseObject is string sourceString)
        {
            return sourceString.Append(characters);
        }
        else throw new PSInvalidOperationException();
    }
    private static string Example() {
        var myString = "Some Value.";
        Console.WriteLine(myString.Append(" and then some more.".ToCharArray()));
        // console output: 
        // Some Value. and then some more.
    }
}

将类型定义加载到 PowerShell 后:

$method = [MyStringExtensions].GetMethod('PSAppend')
Update-TypeData -TypeName System.String -MemberName Append -MemberType CodeMethod -Value $method
# now you can use the method the same way you'd use an extension method in C#
PS:\> $myString = "Some Value."
PS:\> $myString.Append(" and then some more.")
Some value. and then some more.

Documentation for code methods 不太理想。如果您将其构建到模块中并在清单 (.psd1) 引用的 Types.ps1xml 文件中定义 CodeMethod,则需要在显现。 (将它包含为 RequiredAssemblies 是不够的,因为必须在加载类型文件之前加载类型的程序集。)

以下是在 Types.ps1xml 文件中包含此类型定义的方法:

RootModule