3

是否可以在 Windows Phone 7/8 上使用 protobuf-net 序列化/反序列化类型?
我试过下面的代码,似乎不支持构造函数跳过(即 UseConstructor = false)所以我创建了一个无参数构造函数,但反序列化失败,“尝试访问该方法失败:Wp7Tests.ImmutablePoint.set_X(System.Double )"

public class ImmutablePoint
{
    public double X { get; private set; }
    public double Y { get; private set; }
    public ImmutablePoint() {}

    public ImmutablePoint(double x, double y)
    {
        X = x;
        Y = y;
    }
}
public sub Test()
{
        ImmutablePoint pt = new ImmutablePoint(1, 2);
        var model = TypeModel.Create();
        var ptType = model.Add(typeof(ImmutablePoint), false);
        ptType.AddField(1, "X");
        ptType.AddField(2, "Y");
        IsolatedStorageFile store = IsolatedStorageFile.GetUserStoreForApplication();

        using (var stream1 = new IsolatedStorageFileStream("test.bin", FileMode.Create, store))
        {
            try
            {
                model.Serialize(stream1, pt);
            }
            catch (Exception e)
            {                    
                Debugger.Break();
            }
        }
        const double EPSILON = 0.001;
        using (var stream1 = new IsolatedStorageFileStream("test.bin", FileMode.Open, store))
        {
            try
            {
                ImmutablePoint ptDeSer = (ImmutablePoint) model.Deserialize(stream1,null, typeof(ImmutablePoint));
                Debug.Assert(Math.Abs(pt.X - ptDeSer.X) < EPSILON);
                Debug.Assert(Math.Abs(pt.Y - ptDeSer.Y) < EPSILON);
            }
            catch (Exception e)
            {
                Debugger.Break();
            }
        }
}
4

2 回答 2

8

您可以使用代理;这里我用属性装饰它,但也可以手动配置:

[ProtoContract]
public class MutablePoint {
    [ProtoMember(1)] public double X { get; set; }
    [ProtoMember(2)] public double Y { get; set; }

    public static implicit operator MutablePoint(ImmutablePoint value) {
        return value == null ? null : new MutablePoint {X=value.X, Y=value.Y};
    }
    public static implicit operator ImmutablePoint(MutablePoint value) {
        return value == null ? null : new ImmutablePoint(value.X, value.Y);
    }
}

然后告诉模型在看到时使用它ImmutablePoint

var model = TypeModel.Create();
model.Add(typeof(MutablePoint), true);
model.Add(typeof(ImmutablePoint), false).SetSurrogate(typeof(MutablePoint));

序列化程序将根据需要使用运算符在它们之间切换。序列化程序将使用任一implicitexplicit自定义转换运算符。

编辑:像这样反序列化

ImmutablePoint ptDeSer = (ImmutablePoint)model.Deserialize(stream1, null, typeof(ImmutablePoint));
于 2013-04-18T06:34:36.057 回答
2

这在 Windows Phone 上是不可能的。反序列化程序无法访问设置器,并且使用反射访问私有成员会导致 Windows Phone 上的 MemberAccessException。我能想到让这样的事情起作用的唯一方法是创建一个用于序列化的包装器,然后它可以调用 ImmutablePoint 的构造函数。这有点混乱,因为包装类需要 X 和 Y 的公共读写属性。

您还可以查看使用 InternalsVisibleTo 属性来允许序列化程序访问属性(假设您将设置器从私有更改为内部)。不过还是乱七八糟...

于 2013-04-17T21:13:54.837 回答