套接字传输

时间:2011-03-09 14:07:21

标签: java sockets vector

我有几个同时运行的类从服务器获取输入。两者都设置了套接字。

问题是,即使我在类之间传输相同的数据,当数据到达时,它也不一样。

这是发送方法:

        private boolean transferBroadcastData() {
            boolean success = false;

            try {           
                Vector<Client> clientsList =
                    onlineList.getClientsList();

                for (Client client : clientsList) {
                        ObjectOutputStream objectOutputStream =
                            client.getObjectOutputStream();
                            objectOutputStream.writeObject(clientsList);
                            objectOutputStream.flush();
                }


                success = true;

            } catch (Exception ioException) {
                ioException.printStackTrace();
            }

            return success;
        }

这是接收方法:

        while (true) {
                try {
                    Object object = objectInputStream.readObject();

                    if (object instanceof Vector<?>) {
                        String x = "I got it:\n";

                        Vector<Client> clients = (Vector<Client>) object;

                        for(Client c : clients) {
                            x += "\n" + c;
                        }

                        JOptionPane.showMessageDialog(ClientManager.this, x);

                        usersPanel.addClientsToList((Vector<Client>) object);
                    }

                    else if (object instanceof Message)
                        messagePanel.appendText((Message) object);


                } catch (ClassNotFoundException classNotFoundException) {
                    statusPanel.updateStatus("Error reading from socket");
                }
            }

当我在所有客户端上收到Vector时,其中的元素是不同的。它不是我试图转移的Vector的内容。

我错过了什么?

1 个答案:

答案 0 :(得分:1)

使用Serialization / ObjectOutputStream可能会出现问题。当对象被序列化时,ObjectOutputStream会保留内部记录以“优化”序列化过程,因此,如果将同一对象序列化两次,则只需重复ID,假定对象未更改。例如,如果您运行代码:

ObjectOutputStream oos = new ObjectOutputStream();
MyObject[] arr = new MyObject[]{myObject};
oos.write(myObject);
oos.write(arr);

输出可能看起来像(注意不是实际格式只是一个例子):

[[myObject;id=357;foo=bar;baz=36;]]
[[MyObject[];id=358;data={id=357}]]

请注意,数组不会再次序列化对象,而只是已经序列化版本的id。如果您有代码,请更进一步:

ObjectOutputStream oos = new ObjectOutputStream();
oos.write(myObject);
myObject.setBaz(99999);
oos.write(myObject);

你最终会得到:

[[myObject;id=357;foo=bar;baz=36;]]
{id=357}

请注意,即使更改了对象,内部ID也会被序列化。技巧是在编写对象时需要清除ObjectOutputStream的内部状态,这可以通过reset()来完成。来自javadocs:

  

重置将忽略任何状态   已写入流的对象。   状态被重置为与a相同   新的ObjectOutputStream。

因此,如果您采用前面的示例并添加重置:

ObjectOutputStream oos = new ObjectOutputStream();
oos.write(myObject);
myObject.setBaz(99999);
oos.reset();
oos.write(myObject);

您应该获得以下预期行为的输出:

[[myObject;id=357;foo=bar;baz=36;]]
[[myObject;id=358;foo=bar;baz=99999;]]

需要注意的一件事是要注意,第二个序列化现在将被视为与原始序列化不同的对象。根据您传输数据的方式和内部关系,如果对象彼此共享引用并且您希望维护这些关系,则必须注意不要过早地重置流。