5

大量のフィールド/プロパティで初期化されたC#のExpandoObjectがあり、PowerShell環境でこのオブジェクトを使用したいと考えています。PowerShellでそのようなオブジェクトを取得すると、すべてのフィールド/プロパティが表示されるわけではありませんが、(ExpandoObjectsの基になるディクショナリ構造に基づいて)キーと値のペアとして表示されます。

私の実装の目的では、これは非常に問題があり、このキー/値のペアを、そのようなオブジェクトが動作するようにフィールド/プロパティに変換する手段を見つけることができませんでした。ExpandoObjectをObjectにキャストすることもできません。私は何かが足りないのですか?

カスタムメイドDLL(DataCollect.dll)のマージ関数

public static dynamic merge(dynamic obj)
{
    // FIRST  : Get the objects out of the table layout.
    dynamic Data = retrieveObject(obj);

    // SECOND : Check if we're dealing with an array or with a single object.
    bool isCollect = isCollection(Data);

    // THIRD  : Merge objects differently depending on (bool)isCollect.
    // The functions below are merge functions that make use of the ExpandoObject's 
    // underlying Dictionary structure to display it's internal fields/properties.
    if (isCollect)
        return (Object)mergeObjectCollection(Data);
    else
        return (Object)mergeObject(Data);
}

以下に、C#dllをロードしてマージ関数を呼び出すために使用するPowerShellスクリプトを示します。

#Loads in the custom DLL created for this specific project.
[Reflection.Assembly]::LoadFrom("blablbabla/DataCollect.dll")

# Creates a new Client object that handles all communication between the PowerShell module and the
# sncdb-worker at server side.
$client = new-object blablbabla.Sender;
[blablbabla.Config]::Configure("blablbabla/blablbabla.ini")
$client.Connect();

# This functions returns a Host Machine (Virtual or Physical) in object notation to for easy post-processing
# in PowerShell. 
Function SNC-GetHost($hostz = "blablbabla")
{
    $obj = $client.sendMessage([blablbabla.Parser]::getHostIp($hostz)); 
    return ([blablbabla.Merger]::merge($obj)).Value;    
}

そして私のPSコマンドの結果: imagelink



アップデート

C#からPowerShellに正しく変換する方法はまだわかりませんが、HashTablesからPowerShellでオブジェクトを作成する際の小さなトリックを見つけました。重要なのは、Add-Memberコマンドレットを使用することです。このコマンドレットを使用すると、ベースオブジェクト(たとえば、System.Object)の上にオブジェクトを動的に構築できます。

そこで、HashTablesからオブジェクトを再帰的に構築するモジュールを構築しました(プロパティはHashTables(またはExpandoObjects)にもなる可能性があるため、再帰的なメソッドを使用しています)。

#############################################################################
#       PowerShell Module that supplements the DataCollector Library.       #       
#         Generated on: 8/7/2012        Last update: 8/17/2012              #
#############################################################################

function HashToObject($hash)
{
    # Create a placeholder object
    $object = New-Object System.Object;
    # Dynamically add Properties to our Placeholder object from the HashTable
    $hash | foreach { 
        $object | Add-Member NoteProperty $_.Key $_.Value 
    }
    # Search for collections and recursively expand these collections to 
    # objects again.
    $object | Get-Member | 
    foreach { 
        if($_.Definition.StartsWith("System.Dynamic.ExpandoObject")) 
        { 
            Write-Host "Recursively continued on object: " -foregroundcolor DarkGreen -nonewline
            Write-Host $_.Name -foregroundcolor Yellow
            $object.($_.Name) = HashToObject($object.($_.Name))
        }
    }
    return $object;
}

これは実際には機能しますが、いくつかの欠点があります。まず第一に、それは私のタイプ情報を保持しません。(コレクションを除く)すべてがint、bool、floatなどではなく文字列になりました。2番目の問題は、おそらく最良で最もクリーンなソリューションではありません。PowerShellユーザーが低レベルの機能を可能な限り抽象化するために、C#DLL内のすべてを処理することを好みます。

このソリューションは機能するかもしれませんが、それでもより良い実装が必要です。

4

1 に答える 1

5

これまでに見つけた最良のオプションは、コマンドレットを記述するときに PSObject を使用することです。PSObject では、任意の PSVariable オブジェクトを追加して、基本的に同じことを行うことができます。例えば:

while (reader.Read())
{
    var record = new PSObject();
    for (var i = 0; i < reader.FieldCount; i++)
    {
        record.Properties.Add(new PSVariableProperty(
            new PSVariable(reader.GetName(i), reader.GetValue(i))
            ));
    }

    WriteObject(record);
}

PSObject を直接使用することも、ExpandoObject を変換する短いスニペットを記述することもできます。オブジェクトのネストがより深いように見えるシナリオを考えると、変換スニペットがより良い方法かもしれません。

于 2013-03-20T04:08:52.627 に答える