使用javascriptSerializer在powershell 2.0中使用Json循环引用

内容来源于 Stack Overflow,并遵循CC BY-SA 3.0许可协议进行翻译与使用

  • 回答 (1)
  • 关注 (0)
  • 查看 (140)

我在powershell 2.0中编写了一个脚本,目前无法升级到3.0或更高版本。在这个脚本中,我尝试使用此链接中的代码将一些数据序列化为JSON(PowerShell 2.0 ConvertFrom-Json和ConvertTo-Json实现):

function ConvertTo-Json20([object] $item){
    add-type -assembly system.web.extensions
    $ps_js=new-object system.web.script.serialization.javascriptSerializer
    return $ps_js.Serialize($item)
}

我的问题是我以某种方式获得循环引用,我真的不知道为什么。我设置了一小段测试数据,结构在powershell中看起来像这样:

$testRoot = @{
    "id" = "1"
    "children" = @( 
        @{
            "id" = "2"
            "children" = @( 
                @{
                    "id" = "2";
                };
                @{
                    "id" = "3";
                }
            );
        };
        @{
            "id" = "4"
            "children" = @( 
                @{
                    "id" = "5";
                }
            );
        }
    )
}

我知道它看起来很垃圾,但我只需要这种格式。

我需要序列化的结构有更多的层,所以更多的“孩子”,并有它变得奇怪的点。

当我尝试这个:

ConvertTo-Json20 $testRoot

一切正常。结构被解析如下:

{
   "id":"1",
   "children":[
        {
            "id":"2",
            "children":[
               {
                   "id":"2"
               },
               {
                   "id":"3"
               }
            ]
        },
        {
            "id":"4",
            "children":[
               {
                   "id":"5"
               }
            ]
        }
   ]
}

但现在出现了问题。如上所述,结构有更多的层,所以我尝试这只是在数组中设置数据。

ConvertTo-Json20 @($testRoot)

但它不起作用我只是收到一条错误消息:

Exception in method "Serialize" with 1 argument(s):  
"While serializing an object of type "System.Management.Automation.PSParameterizedProperty" a circular reference was discovered."
At C:\Users\a38732\Desktop\Temp.ps1:34 symbol:28
+     return $ps_js.Serialize <<<< ($item)
+ CategoryInfo          : NotSpecified: (:) [], MethodInvocationException
+ FullyQualifiedErrorId : DotNetMethodException

(我翻译了德语的错误信息,所以在英文版本中可能会有一些不同的词......)

提问于
用户回答回答于

一个问题是JavaScriptSerializer类本身的使用。截至此日期,文档本身承认它不应该用于序列化或反序列化JSON。去引用:

Json.NET应该用于序列化和反序列化。

如果你能够使用像Json.NET这样的第三方库,这里有一个简单的函数,可以根据OP中的数据结构执行所需的操作:

function ConvertTo-JsonNet {
   [CmdletBinding()]
    param(
        [Parameter(Mandatory)] $object,
        [Parameter(Mandatory)] [string]$jsonNetPath,
        [switch]$indent,
        [switch]$preserveReferencesHandling
    )
    Add-Type -Path $jsonNetPath;

    $formatting = if ($indent.IsPresent) { [Newtonsoft.Json.Formatting]::Indented; }
    else { [Newtonsoft.Json.Formatting]::None; }

    $settings = New-Object Newtonsoft.Json.JsonSerializerSettings;
    if ($preserveReferencesHandling.IsPresent) { 
        $settings.PreserveReferencesHandling = [Newtonsoft.Json.PreserveReferencesHandling]::Objects;
    }

    [Newtonsoft.Json.JsonConvert]::SerializeObject($object, $formatting, $settings);
}

简单的用法,假设Newtonsoft.Json.dll与脚本位于同一目录中:

$dllPath = Join-Path $PSScriptRoot 'Newtonsoft.Json.dll';
ConvertTo-JsonNet @($testRoot) $dllPath;

输出:

[{"id":"1","children":[{"id":"2","children":[{"id":"2"},{"id":"3"}]},{"id":"4","children":[{"id":"5"}]}]}]

您可以从nuget包项目站点手动下载.dll 。它有一个.nupkg文件扩展名,但它是一个压缩存档,因此将扩展名重命名为.zip您已设置。在lib子目录中有.NET版本从2.0到4.5的.dll文件。

扫码关注云+社区

领取腾讯云代金券

年度创作总结 领取年终奖励