4

我们的任务非常简单,我们有一个对象图,其中每个对象(IDItem)都有一个唯一的 ID。对象图存在两次,分别在客户端和服务器机器上。

现在我们将一些可序列化的命令传递给服务器。该命令具有一些 IDItems 作为字段。IDItem 实现了 ISerializable 接口,并且只将它们的 ID 存储在 SerializationInfo 中。像:

// The method called when serializing a IDItem.
void GetObjectData(SerializationInfo info, StreamingContext context)
{
    // Instead of serializing this object, just save the ID
    info.AddValue("ID", this.GetID());
}

问题是,我们如何将现有对象分配给反序列化器创建的实例?显然,ISerializable 构造函数中的以下内容不起作用,因为“this”标识符是只读的:

//does not work   
protected IDItem(SerializationInfo info, StreamingContext context)
{
    this = GlobalObject.GetIDItem(info.GetString("ID"));
}

那么知道我们如何将现有对象分配给反序列化对象吗?

最好的问候, 塔尔姆

4

2 回答 2

5

可以通过创建一个实现IObjectReference并执行假反序列化的代理对象来做到这一点。

(您的代理对象需要同时存在于客户端和服务器上,我想您的类型版本控制等也需要在两者上完全相同。)

[Serializable]
public class Example : ISerializable
{
    // ...

    void ISerializable.GetObjectData(
        SerializationInfo info, StreamingContext context)
    {
        info.SetType(typeof(ExampleDeserializationProxy));
        info.AddValue("ID", this.GetID());
    }
}

// ...

[Serializable]
public class ExampleDeserializationProxy : IObjectReference, ISerializable
{
    private readonly int _id;

    private ExampleDeserializationProxy(
        SerializationInfo info, StreamingContext context)
    {
        _id = info.GetInt32("ID");
    }

    object IObjectReference.GetRealObject(StreamingContext context)
    {
        return GlobalObject.GetIDItem(_id);
    }

    void ISerializable.GetObjectData(
        SerializationInfo info, StreamingContext context)
    {
        throw new NotSupportedException("Don't serialize me!");
    }
}
于 2010-07-09T15:29:59.620 回答
1

你不能直接那样做。

与其序列化整个对象但只包含一个字段,为什么不直接发送 ID 并在客户端和/或服务器上进行查找?如果我理解正确,服务器上已经存在该对象的实例。您应该没有理由需要传递该对象的剥离版本来再次查找相同的对象。

如果 ID 是唯一的并且与对象一起存储在某个集合中,则将 ID 作为字符串或任何数据类型传递,应该足以在现有对象集合中执行查找。

于 2010-07-09T14:18:31.140 回答