我在Socket中使用了ObjectStream来传输类。
客户端在同一类中传输两个不同的数据,但是服务器获取的第二个数据与第一个数据相同。它很奇怪!

这是我的客户代码:

public Client()
{
        MessageClass messageobject=new MessageClass("login");
        messageobject.SetLoginUserInfo("18580409","12345","magicgiant");

    try
    {
        client=new Socket("localhost",1234);
        System.out.println("Connected!");
        ObjectOutputStream out=new ObjectOutputStream(client.getOutputStream());
        out.writeObject(messageobject);
        out.flush();
        System.out.println(1);

        messageobject.inquire=true;
        messageobject.SetLoginUserInfo("122131","21312","dfsd");
        out.writeObject(messageobject);
        out.flush();
        System.out.println(2);
    }
    catch(Exception e)
    {
        System.out.println(e);
    }
}


这是我的服务器:

public void run()
{
  try
  {
     is=new ObjectInputStream(client.getInputStream());
  }
  catch (Exception ex)
  {
     System.out.println(ex);
  }

  while(true){

   try
   {
      MessageClass messageobject = (MessageClass)is.readObject();
      System.out.println(messageobject.GetLoginId()+messageobject.GetLoginPassword());
      idSocketItem = new IdSocket(messageobject.GetLoginId(),client,messageobject);
      s.idSocketList.addElement(idSocketItem);
    }
    catch (Exception ex)
    {
       System.out.println(ex);
     }

   }
}


服务器中的System.out应该是

1858040912345
12213121312


但是真正的结果是

1858040912345
1858040912345


我试图取消flush(),但是它不起作用。
问题出在哪里?

最佳答案

问题是您要编写两次相同的引用:

out.writeObject(messageobject);
out.flush();
System.out.println(1);

messageobject.inquire=true;
messageobject.SetLoginUserInfo("122131","21312","dfsd");
out.writeObject(messageobject);


ObjectOutputStream缓存您已经编写了该引用的事实,并且仅发出一个令牌来引用先前的值。要考虑的三个选项:


(首选,IMO):创建一个新对象,而不是修改现有对象
呼叫writeUnshared()而不是writeObject()



  此方法与writeObject相同,除了它始终将给定对象写为流中新的唯一对象(与指向先前序列化实例的反向引用相反)。
  写入第一个对象后,在流上调用reset()



  重置将忽略任何已写入流的对象的状态。将状态重置为与新的ObjectOutputStream相同。流中的当前点被标记为reset,因此相应的ObjectInputStream将在同一点被重置。先前写入流中的对象将不会被视为已经存在于流中。它们将再次写入流中。



我更喜欢第一个版本,因为从逻辑上讲您有两个不同的消息-因此它们应该是两个不同的对象。

另外,我强烈建议您开始遵循Java命名约定,并将所有字段设为私有以鼓励封装。

09-10 07:56
查看更多