1

我已经建立了一个 android TCP 连接,其中服务器是用 java 编写的,客户端是用 android 编写的。它基本上是一个服务器客户端聊天。代码一切正常,运行良好。问题是,当我关闭 android 客户端时,服务器忘记了客户端并且不再检索该客户端,但服务器和客户端都需要重新启动。我希望当我关闭客户端但服务器仍在运行时再次运行客户端,然后服务器应该搜索他最近的客户端并恢复服务器客户端聊天。我正在提供我的代码。请有人帮助尽快修改我的代码。谢谢

TCPServer.java

import javax.swing.*;
import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

/**
* The class extends the Thread class so we can receive and send messages at the same   time */
public class TCPServer extends Thread {

public static final int SERVERPORT = 4444;
private boolean running = false;
private PrintWriter mOut;
private OnMessageReceived messageListener;

public static void main(String[] args) {

    //opens the window where the messages will be received and sent
    ServerBoard frame = new ServerBoard();
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.pack();
    frame.setVisible(true);

}

/**
 * Constructor of the class
 * @param messageListener listens for the messages
 */
public TCPServer(OnMessageReceived messageListener) {
    this.messageListener = messageListener;
}


/**
 * Method to send the messages from server to client
 * @param message the message sent by the server
 */
public void sendMessage(String message){
    if (mOut != null && !mOut.checkError()) {
        mOut.println(message);
        mOut.flush();
    }
}

@Override
public void run() {
    super.run();

    running = true;

    try {
        System.out.println("S: Connecting...");

  //create a server socket. A server socket waits for requests to come in over network.
        ServerSocket serverSocket = new ServerSocket(SERVERPORT);


        Socket client = serverSocket.accept();
        System.out.println("S: Receiving...");

        try {

            //sends the message to the client
   mOut = new PrintWriter(new BufferedWriter(new    OutputStreamWriter(client.getOutputStream())), true);

            //read the message received from client
            BufferedReader in = new BufferedReader(new InputStreamReader(client.getInputStream()));

            //in this while we wait to receive messages from client (it's an infinite loop)
            //this while it's like a listener for messages
            while (running) {
                String message = in.readLine();

                if (message != null && messageListener != null) {
                    //call the method messageReceived from ServerBoard class
                    messageListener.messageReceived(message);
                }
            }

        } catch (Exception e) {
            System.out.println("S: Error");
            e.printStackTrace();
        } finally {
            client.close();
            System.out.println("S: Done.");
        }

    } catch (Exception e) {
        System.out.println("S: Error");
        e.printStackTrace();
    }

}


//Declare the interface. The method messageReceived(String message) will must be implemented in the ServerBoard
//class at on startServer button click
public interface OnMessageReceived {
    public void messageReceived(String message);
} }
4

2 回答 2

0

我假设某种聊天应用程序 - 当用户点击他/她的 Android 设备上的应用程序时 - 将最后的消息显示为历史记录。

识别客户的部分是你必须从设计方面考虑的事情,考虑到数据隐私等问题。让我解释一些替代方案:

  • 通过地址识别客户:这是您可以使用的字符串client.getInetAddress().getHostAddress()。由于 IP 地址可以在 DHCP 环境中重复使用,并且 Android 设备的主机名是由 Android 定义的,用户无法更改,因此这种识别方法可能是一种好方法,但不一定总是同一个客户端。当另一个 Android 设备被意外识别并获取实际另一个用户的历史记录时,您的应用会发生什么情况?
  • 您可能希望将设备的 IMEI 作为唯一标识符传输到服务器,并使用它来识别用户/应用程序。有关如何使用的信息,请参阅TelephonyManagergetDeviceId()
  • 使用上面的两个标识,您可以在不让用户知道的情况下识别用户/设备/应用程序,您这样做了。根据您的应用程序环境的敏感程度(例如公共德语应用程序与您的私人小助手应用程序),这可能是一个法律问题。因此,这里的出路是创建用户帐户,让用户注册、登录、使用您的应用程序并注销。然后,您可以在用户注册时签署免责声明。当然,这是迄今为止最复杂的解决方案。

最后,您需要存储消息的历史记录。我将扩展该行messageListener.messageReceived(message);sendMessage(String message)保存消息的方法,例如在每个客户端的文件中。

于 2013-09-12T06:54:34.737 回答
0

我认为关键是您如何识别客户端,以便一旦它回来,您希望服务器能够记住它。我没有在服务器代码中看到它跟踪客户端的逻辑。

话虽如此,识别 TCP 客户端的一种方法是使用客户端的 IP 地址和端口号。如果我们关闭那个 TCP 客户端并重新启动它,客户端的端口号可能无法得到保证,因为大多数 TCP 客户端不调用 bind() 并且只是使用 connect() 来获取临时端口号。或者,您可以使用应用程序层逻辑(记住应用程序位于 TCP 之上),其中客户端可以使用唯一标识符,或者可以要求服务器为其提供唯一标识符。因此,当客户端返回时,它会告诉服务器它有一个标识符并且它希望恢复该连接。这样,服务器就可以将新的 fd 与它之前发布的标识符相关联。请注意,如果服务器发出标识符,则更容易保证标识符的唯一性——例如,

于 2013-09-12T05:49:09.447 回答