gpt4 book ai didi

c# - protobuf-net:检测到可能的递归

转载 作者:太空狗 更新时间:2023-10-30 00:55:17 26 4
gpt4 key购买 nike

我在尝试序列化对象图(不是很深)时遇到异常。有意义的部分是这样的:

[ERROR] FATAL UNHANDLED EXCEPTION: ProtoBuf.ProtoException: Possible recursion d etected (offset: 5 level(s)): red at ProtoBuf.ProtoWriter.CheckRecursionStackAndPush (object) <0x00127> at ProtoBuf.ProtoWriter.StartSubItem (object,ProtoBuf.ProtoWriter,bool) <0x0002f>

该图表示文件/目录结构,我的模型(简化)如下所示:

[ProtoContract] 
[ProtoInclude(100, typeof(PackageDirectory))]
[ProtoInclude(200, typeof(PackageFile))]
public abstract class PackageMember
{
[ProtoMember(1)]
public virtual string Name { get; protected set; }

[ProtoMember(2, AsReference=true)]
public PackageDirectory ParentDirectory { get; protected set; }
}

[ProtoContract]
public class PackageDirectory : PackageMember
{
[ProtoMember(3)]
private Dictionary<string, PackageMember> _children;

public PackageDirectory()
{
_children = new Dictionary<string, PackageMember>();
}

public PackageDirectory (string name, PackageDirectory parentDirectory)
: this()
{
this.ParentDirectory = parentDirectory;
this.Name = name;
}

public void Add (PackageMember member)
{
_children.Add(member.Name, member);
}
}

[ProtoContract]
public class PackageFile : PackageMember
{
private Stream _file;
private BinaryReader _reader;

private PackageFile()
{}

public PackageFile (string name, int offset, int length, PackageDirectory directory, Stream file)
{
this.Name = name;
this.Length = length;
this.Offset = offset;
this.ParentDirectory = directory;

_file = file;
_reader = new BinaryReader(_file);
}

[OnDeserialized]
protected virtual void OnDeserialized(SerializationContext context)
{
var deserializationContext = context.Context as DeserializationContext;

if (deserializationContext != null)
{
_file = deserializationContext.FileStream;
_reader = new BinaryReader(_file);
}
}

[ProtoMember(3)]
public int Offset { get; private set; }

[ProtoMember(4)]
public int Length { get; private set; }
}

这棵树的深度接近 10-15 层,小于 ProtoBuf.ProtoWriter.RecursionCheckDepth 值 (25)。 (所以也许这是一个错误?)使用的 protobuf-net 版本 是从 trunk v2 (rev 491) 编译而来的。

实际上,我通过修改 protobuf-net 代码解决了这个问题。我将 ProtoBuf.ProtoWriter.RecursionCheckDepth 的值更改为 100,一切似乎都正常。

问题是是否有任何“真正的”方法可以在不修改 protobuf 代码的情况下序列化这种图?这种行为是正确的还是错误?

我的平台是 Windows 7 Professional 64 位上的 Mono-2.10-8

附言我还发现,如果我使用以下代码反序列化,我应该将 PackageDirectory 无参数构造函数公开。

var value = new PackageDirectory();
RuntimeTypeModel.Default.Deserialize(ms, value, typeof(PackageDirectory), new SerializationContext {
Context = new DeserializationContext {
FileStream = _file,
}});

这是另一个主题,但已通过提供的代码进行了很好的说明。我认为在这种情况下应该允许声明私有(private)构造函数,因为现在行为不同于 Serializer.Deserialize(...)。

最佳答案

仅当在数据中看到相同的引用(在同一路径中两次)时才会抛出此异常,并且仅在深度至少为时启用跟踪 递归检查深度。这立即让我怀疑引用的 10-15 深度限制,尽管 protobuf 处理级别的情况不一定相当与您计算的相同。将这个数字提高到 100 对我来说没有任何意义 - 事实上,这个 RecursionCheckDepth 的存在纯粹是一种优化,以限制“典型”图形所涉及的工作量,仅如果它开始深入观察,则启用更严格的检查。

不过,我注意到这可能还暗示了基于继承的处理中的一些细微错误,可能也与 AsReference 相关。我广泛且不断地使用 protobuf-net,我还没有看到这样的问题。如果您有可复制的复制品,我非常希望看到它。

关于c# - protobuf-net:检测到可能的递归,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10070710/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com