2011-12-07 14 views
5

私は、powershell 2.0でWebサービスで使用するために、ハッシュテーブルをjsonオブジェクトに変換しようとしています。hastableをpowershellのjson文字列に変換するにはどうしたらいいですか?

$testhash = @{ 
    Name = 'John Doe' 
    Age = 10 
    Amount = 10.1 
    MixedItems = (1,2,3,"a") 
    NestedHash = @{ 
     nestedkey = "nextedvalue" 
    } 
} 

function toJson($obj){ 

    $ms = New-Object IO.MemoryStream 
    $type = $obj.getType() 
    [Type[]]$types = ($obj | select -expand PsTypeNames | Select -unique) + [type]'System.Management.Automation.PSObject' 
    $js = New-Object System.Runtime.Serialization.Json.DataContractJsonSerializer $type, $types, ([int]::MaxValue), $false, $null, $false 
    $js.writeObject($ms, $obj) | out-null 
    $utf8.GetString($ms.ToArray(), 0, $ms.Length) 
    $ms.Dispose() | out-null 
} 

toJson $testhash 
'[{"Key":"Name","Value":"John Doe"},{"Key":"Age","Value":10},{"Key":"Amount","Value":10.1},{"Key":"NestedHash","Value":[{"__type":"KeyValuePairOfanyTypeanyType:#System.Collections.Generic","key":"nestedkey","value":"nextedvalue"}]},{"Key":"MixedItems","Value":[1,2,3,"a"]}]' 

私は型情報を抑制する必要がある方法でDataContractJsonSerializer constructorを使用していますが、それは明らかではありません。私はキーとバリューのペアを抽出することによっても面白いですが、それをしないようにしたいと思います。私は間違って何をしていますか?

答えて

3

私は以下のようにhereからスクリプトを適応:

$testhash = @{ 
    Name = 'John Doe' 
    Age = 10 
    Amount = 10.1 
    MixedItems = (1,2,3,"a") 
    NestedHash = @{ 
     nestedkey = "nextedvalue" 
    } 
} 

function Read-Stream { 
PARAM(
    [Parameter(Position=0,ValueFromPipeline=$true)]$Stream 
) 
process { 
    $bytes = $Stream.ToArray() 
    [System.Text.Encoding]::UTF8.GetString($bytes,0,$bytes.Length) 
}} 

function New-Json { 
[CmdletBinding()] 
param([Parameter(ValueFromPipeline=$true)][HashTable]$InputObject) 
begin { 
    $ser = @{} 
    $jsona = @() 
} 
process { 
    $jsoni = 
    foreach($input in $InputObject.GetEnumerator() | Where { $_.Value }) { 
     if($input.Value -is [Hashtable]) { 
     '"'+$input.Key+'": ' + (New-JSon $input.Value) 
     } else { 
     $type = $input.Value.GetType() 
     if(!$Ser.ContainsKey($Type)) { 
      $Ser.($Type) = New-Object System.Runtime.Serialization.Json.DataContractJsonSerializer $type 
     } 
     $stream = New-Object System.IO.MemoryStream 
     $Ser.($Type).WriteObject($stream, $Input.Value) 
     '"'+$input.Key+'": ' + (Read-Stream $stream) 
     } 
    } 

    $jsona += "{`n" +($jsoni -join ",`n")+ "`n}" 
} 
end { 
    if($jsona.Count -gt 1) { 
     "[$($jsona -join ",`n")]" 
    } else { 
     $jsona 
    } 
}} 

$testHash | New-Json 
+0

これはノーシリアライズ関数は、それ自身で行う必要があります何かのように思えますか?どちらの方法でも、これはほぼ完全に動作します。ネストされたハッシュの2つのレベルで失敗します。 – reconbot

+0

@wizard - 単純なハッシュテーブルだと思います。私はJSON解析にリン​​クしているようなスクリプトが既にあります。 – manojlds

+0

私はhttp://poshcode.org/2930の問題を持っていますが、十分にうまく機能している使用して終了 - 1日のPowerShell 3.0利用できるようになりますと、私はこれらの無駄な時間を振り返ると動揺になりますが – reconbot

16

[OK]を、私はちょうどここv3の同等物を投げるますのでmanojldsがV2のために答えて:

PS> @{name="oisin"; age=37} | convertto-json 
{ 
    "age": 37, 
    "name": "oisin" 
} 

かなりクリーナー、 右?

のPowerShell 3.0 CTP2:http://www.microsoft.com/download/en/details.aspx?id=27548

+2

私はPowerShellのV3を使用することがしたいです= ) – reconbot

+0

とてもきれいで簡単! – lantrix

+0

この_shell_にかなりの_power_があります:) – raghav710

関連する問題