我正在用Powershell2.0编写脚本,现在不可能升级到3.0或更高版本。在此脚本中,我尝试使用此链接中的代码(PowerShell 2.0 ConvertFrom-Json and ConvertTo-Json implementation)将一些数据序列化为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
(我将错误消息从德语翻译过来,因此在英语版本中可能会有一些不同的单词...)
发布于 2018-07-18 07:05:02
其中一个问题是JavaScriptSerializer
类本身的使用。截至此日期,documentation itself concedes it should not be used to serialize nor deserialize 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 package project site手动下载.dll。它有一个.nupkg
文件扩展名,但它是一个压缩的归档文件,所以将扩展名重命名为.zip
,您就设置好了。在lib
子目录中,有用于从2.0到4.5的.NET版本的.dll文件。
https://stackoverflow.com/questions/39161783
复制相似问题