gpt4 book ai didi

通过套接字缓存 Java 对象序列化/反序列化

转载 作者:行者123 更新时间:2023-12-02 00:08:31 25 4
gpt4 key购买 nike

我有一个 Java 客户端/服务器应用程序,它通过序列化/反序列化对象进行通信。通常,客户端发送RequestObj,服务器使用ResponseObj 进行响应。服务器也发送心跳RequestObj,客户端响应心跳ResponseObj。

//Sample RequestObj
public class RequestObj implements Serializable {
private static final long serialVersionUID = xxxx; //auto generated
int type; //TYPE_HEARTBEAT
long time;
...

public void setTime() {
this.time=System.currentTimeMillis();
}
}

当从服务器向每个客户端发送心跳请求时,它重用相同的RequestObj(每个连接),设置当前时间,然后发送到客户端。我期待收到包含当前时间的 RequestObj。但是,似乎在客户端反序列化的 RequestObj 总是给我错误的时间。事实上,客户端收到的RequestObj与第一个心跳RequestObj完全相同(相同的对象id/地址)。但不更新的时间?

//Sample Server side code sending heartbeat
public class ServerSocketWrapper {
Socket socket;
ObjectOutputStream oos;
...
RequestObj heartbeat = New RequestObj(TYPE_HEARTBEAT);

public void sendHeartBeat() {
heartbeat.setTime();
logger.info("HeartBeat {} {}", heartbeat.time, formatFullTime(heartbeat.time));
oos.writeObject(heartbeat);
oos.flush();
}
}

这是一些示例客户端代码:

//Sample Client side code receiving heartbeat
public class ClientSocketWrapper {
Socket socket;
ObjectInputStream ois;
long heartbeatTime;
...

public void run() {
while(true) {
Object obj = ois.readObject();
if (obj instanceof RequestObj) {
RequestObj req = (RequestObj) obj;
if (req.type == TYPE_HEARTBEAT) {
heartbeatTime = req.time;
logger.info("Received heartbeat {} {} {}", heartbeatTime, formatFullTime(heartbeatTime), req);
sendResponse(new ResponseObj(req.id, TYPE_HEARTBEAT));
}
}
...
}
}
}

这是日志

logs on the server
16:33:56.186 [pool-2-thread-2] INFO ServerSocketWrapper - HeartBeat 1352842436186 2012-11-13 16:33:56.186
16:34:56.185 [pool-2-thread-2] INFO ServerSocketWrapper - HeartBeat 1352842496185 2012-11-13 16:34:56.185
16:35:56.185 [pool-2-thread-1] INFO ServerSocketWrapper - HeartBeat 1352842556185 2012-11-13 16:35:56.185

logs on the client
16:34:08.510 [server:port] INFO ClientSocketWrapper - Received heartbeat 1352842436186 2012-11-13 16:33:56.186 RequestObj@8497904
16:35:06.758 [server:port] INFO ClientSocketWrapper - Received heartbeat 1352842436186 2012-11-13 16:33:56.186 RequestObj@8497904
16:36:10.303 [server:port] INFO ClientSocketWrapper - Received heartbeat 1352842436186 2012-11-13 16:33:56.186 RequestObj@8497904

为什么客户端收到的时间没有更新? JVM 是否进行某种对象缓存?服务器运行jdk 1.7.07 64b,而客户端运行jdk 1.6.31 64b。

最佳答案

如果使用相同的ObjectOutputStream多次写入单个对象,则ObjectOutputStream将识别出它已经看到该对象并将其编码为对现有对象的引用目的。这就是序列化能够保留引用别名的方式。

如果您不希望跨消息进行引用跟踪,请为每个新消息构造一个新的 ObjectOutputStreamObjectInputStream 对(或 #reset ObjectOutputStream#readUnshared 来自 ObjectInputStream)。如果您不执行其中任何一项,而是从不修改和重用消息对象,则不会出现修改不可见的问题,但您会泄漏资源,因为 ObjectInputStream 将对它反序列化过的每个对象都有一个强引用。

关于通过套接字缓存 Java 对象序列化/反序列化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13369788/

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