有 Java 编程相关的问题?

你可以在下面搜索框中键入要查询的问题!

java使用TCP over socket从服务器端向客户端发送ArrayList

我正在尝试将对象从服务器端发送到客户端,但找不到问题。以下是我在客户端遇到的错误:

java.io.StreamCorruptedException: invalid type code: 43
at java.io.ObjectInputStream.readObject0(Unknown Source)
at java.io.ObjectInputStream.readObject(Unknown Source)
at connection.MainClient.doAll(MainClient.java:56)
at connection.TestScreen$2.actionPerformed(TestScreen.java:82)
at javax.swing.AbstractButton.fireActionPerformed(Unknown Source)
at javax.swing.AbstractButton$Handler.actionPerformed(Unknown Source)
at javax.swing.DefaultButtonModel.fireActionPerformed(Unknown Source)
at javax.swing.DefaultButtonModel.setPressed(Unknown Source)
at javax.swing.plaf.basic.BasicButtonListener.mouseReleased(Unknown Source)
at java.awt.Component.processMouseEvent(Unknown Source)
at javax.swing.JComponent.processMouseEvent(Unknown Source)
at java.awt.Component.processEvent(Unknown Source)
at java.awt.Container.processEvent(Unknown Source)
at java.awt.Component.dispatchEventImpl(Unknown Source)
at java.awt.Container.dispatchEventImpl(Unknown Source)
at java.awt.Component.dispatchEvent(Unknown Source)
at java.awt.LightweightDispatcher.retargetMouseEvent(Unknown Source)
at java.awt.LightweightDispatcher.processMouseEvent(Unknown Source)
at java.awt.LightweightDispatcher.dispatchEvent(Unknown Source)
at java.awt.Container.dispatchEventImpl(Unknown Source)
at java.awt.Window.dispatchEventImpl(Unknown Source)
at java.awt.Component.dispatchEvent(Unknown Source)
at java.awt.EventQueue.dispatchEventImpl(Unknown Source)
at java.awt.EventQueue.access$200(Unknown Source)
at java.awt.EventQueue$3.run(Unknown Source)
at java.awt.EventQueue$3.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
at java.awt.EventQueue$4.run(Unknown Source)
at java.awt.EventQueue$4.run(Unknown Source)
at java.security.AccessController.doPrivileged(Native Method)
at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
at java.awt.EventQueue.dispatchEvent(Unknown Source)
at java.awt.EventDispatchThread.pumpOneEventForFilters(Unknown Source)
at java.awt.EventDispatchThread.pumpEventsForFilter(Unknown Source)
at java.awt.EventDispatchThread.pumpEventsForHierarchy(Unknown Source)
at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
at java.awt.EventDispatchThread.run(Unknown Source)

以下是服务器端方法: 无限循环中的separate方法只是从文件中读取并将用户名和密码解析为ArrayList,ArrayList是服务器类中的实例变量

    public void doStuff() throws Exception
{
    ServerSocket serverSocket = new ServerSocket(5001);
    Socket clientSocket = serverSocket.accept();
    PrintWriter out = new PrintWriter(clientSocket.getOutputStream(), true);
    BufferedReader in = new BufferedReader(
            new InputStreamReader(
            clientSocket.getInputStream()));
    ObjectOutputStream objectOutput = new ObjectOutputStream(clientSocket.getOutputStream());
    String inputLine;
    out.println("Connected");
    while(true)
    {
        seperate();
        while ((inputLine = in.readLine()) != null) 
        {
            if(inputLine.equalsIgnoreCase("users"))
                objectOutput.writeObject(getUserNames());
            else
                if(inputLine.equalsIgnoreCase("pass"))
                    objectOutput.writeObject(getPassWords());
                else
                    if(inputLine.equalsIgnoreCase("stop"))
                        objectOutput.reset();
        }
    }
}

下面是从服务器请求信息的客户端:

public boolean doAll(String name, String pass) throws Exception
{
    try 
    {
        kkSocket = new Socket("PC", 5001);
        out = new PrintWriter(kkSocket.getOutputStream(), true);
        in = new BufferedReader(new InputStreamReader(kkSocket.getInputStream()));
    } 
    catch (UnknownHostException e) 
    {
        System.err.println("Don't know about host: PC.");
        System.exit(1);
    }
    catch (IOException e) 
    {
        System.err.println("Couldn't get I/O for the connection to: PC.");
        System.exit(1);
    }

    ObjectInputStream objectInput = new ObjectInputStream(kkSocket.getInputStream());

    out.println("user");
    Object obj = objectInput.readObject();
    users = (ArrayList<String>)obj;
    out.println("pass");
    obj = objectInput.readObject();
    this.pass = (ArrayList<String>)obj;
    out.println("stop");
    objectInput.close();
    out.close();
    in.close();
    kkSocket.close();

    if(userIsRegistered(name,pass))
        return true;
    return false;
}

我在学习服务器socket和socket方面是新手,但我在这里尝试完成的是以下内容。当我按下另一个类中的按钮时,我会执行以下操作:

MainClient b = new MainClient();
login = b.doAll(userField.getText(),passField.getText());
Login is obviously a boolean.

几乎是一个通过服务器到cleint的登录系统,当按下login按钮时,它会调用一个客户端来连接到该服务器,并获取用户列表,并传递到存储在指定目录中的文本文件中的服务器。然后,当客户机收到此信息时,它会检查是否可能是我从两个文本字段中获取的字符串中的用户,并返回true或false。然后关闭服务器的socket,每次都应该重新连接,对吗

我的主要问题是如何修复它不断抛出的错误


共 (2) 个答案

  1. # 1 楼答案

    您的代码流程如下所示:

    1. 服务器已启动,正在等待与客户端的连接
    2. 服务器使用serverSocket.accept()接受来自客户端的连接
    3. 服务器通过out.println("Connected")使用PrintWriter对象向客户端发送String消息(“已连接”)
    4. 客户端正在使用ObjectInputStream对象读取该消息,这会导致流不匹配,从而违反内部一致性检查。这就是为什么在客户端抛出StreamCorruptedException。因此,您应该使用上面创建的BufferedReader对象,而不是使用ObjectInputStream来读取服务器发送的普通消息

    因此,在客户端,您的代码应该修改为:

    ObjectInputStream objectInput = new ObjectInputStream(kkSocket.getInputStream());
    String message = in.readLine();//use BufferedReader to read the plain message coming from Server.
    System.out.println("Message from server: "+message);
    out.println("user");
    Object obj = objectInput.readObject();
    
  2. # 2 楼答案

    不能将InputStreamReader与字节输入流一起使用。它需要输入流中的字符/字符串。试试ObjectInputStream

    只是一个提示:输入流类型应该在服务器端和客户端都匹配。你不能把它们混在一起。在您正在使用ObjectOutputStream的服务器上,因此应在客户端上使用相应的ObjectInputStream