java - 通过 tcp 套接字从 java 服务器向 Android 客户端返回消息

标签 java server sockets client tcp

我正在尝试建立一个非常简单的客户端/服务器连接。我有 Android 应用程序作为客户端并尝试

1) 将消息传递给PC端的Java程序,然后

2) 向android客户端返回一条消息。

第一部分工作正常。问题在于将消息从服​​务器返回到客户端。

服务器代码(Java):

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;


public class Main {

    private final static Integer IN_PORT = 4444; 

    private static ServerSocket serverSocket = null;
    private static Socket sktIn;
    private static InputStreamReader inputStreamReader;
    private static BufferedReader bufferedReader;
    private static String message;

    private static PrintWriter printWriter;

    public static void main(String[] args) {

        try {
            serverSocket = new ServerSocket(IN_PORT);  //Server socket
            System.out.println("Server started. Listening to the port " + IN_PORT);        
        } catch (IOException e) {
            System.out.println("Could not listen on port: " + IN_PORT);
        }


        while (true) {
         try {
          sktIn = serverSocket.accept();   //accept the client connection
                inputStreamReader = new InputStreamReader(sktIn.getInputStream());
                bufferedReader = new BufferedReader(inputStreamReader); //get the client message
                message = bufferedReader.readLine();

                printWriter = new PrintWriter(sktIn.getOutputStream(), true);
                printWriter.write("Returned back \n");  //write the message to output stream
                printWriter.flush();
                printWriter.close();             
                inputStreamReader.close();               
                sktIn.close();

                System.out.println(message);                               

            } catch (IOException ex) {
                System.out.println("Problem in message reading/sending.");
                ex.printStackTrace();
            }

        }

    }
}

客户端主要 Activity (Android):

import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;

public class SimpleClientActivity extends Activity {

    private final Integer OUT_PORT = 4444;
    private final String S_IP = "192.168.1.104"; 

    private EditText textField;
    private Button button;
    private String message;


    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);    

        textField = (EditText) findViewById(R.id.editText1); //reference to the text field
        button = (Button) findViewById(R.id.button1);   //reference to the send button

        //Button press event listener
        button.setOnClickListener(new View.OnClickListener() {

            public void onClick(View v) {

                message = textField.getText().toString(); //get the text message on the text field
                textField.setText("");      //Reset the text field to blank

                new ConnectClient(message, S_IP, OUT_PORT, getApplicationContext()).execute(); 
            }
         });
    }
}

用于连接的单独 AsyncTask 类:

import android.content.Context;
import android.os.AsyncTask;
import android.widget.Toast;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.Socket;
import java.net.UnknownHostException;

public class ConnectClient extends AsyncTask<String, Void, String> {

    private Socket socket;
    private PrintWriter printWriter;

    private String param;   
    private Context context;
    private Integer PORT;
    private String IP;

    private static InputStreamReader inputStreamReader;
    private static BufferedReader bufferedReader;
    private static String message;


 public ConnectClient(String par, String ip, Integer prt, Context ctx){
  super();
     this.context = ctx;
     this.param = par;
     this.PORT = prt;
     this.IP = ip;
 }

 @Override
    public void onPreExecute() {
        Toast.makeText(context, "start " + param, Toast.LENGTH_SHORT)
           .show();
    }

 @Override
    protected String doInBackground(String... params) {

        try {
                socket = new Socket(IP, PORT);  //connect to server

                printWriter = new PrintWriter(socket.getOutputStream(), true);
                printWriter.write(param);  //write the message to output stream

                printWriter.flush();
                printWriter.close();

                socket = new Socket(IP, PORT); // second connection to server     
                inputStreamReader = new InputStreamReader(socket.getInputStream());
                message = "after isr";
                bufferedReader = new BufferedReader(inputStreamReader); //get the client message
                message = bufferedReader.readLine();
                inputStreamReader.close();          
                socket.close();   //closing the connection

            } catch (UnknownHostException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }

        return message;
    }

    @Override
    protected void onPostExecute(String result) {
        Toast.makeText(context, result, Toast.LENGTH_SHORT).show()   
    }

    @Override
    protected void onProgressUpdate(Void... values) {
        Toast.makeText(context, "In progress", Toast.LENGTH_SHORT).show();
    }
}

Java 程序执行正常,Android 应用程序运行没有任何问题,但最终结果并不如人意。

如果我删除第二个 socket = new Socket(IP, PORT);//第二次连接到服务器 ,然后服务器可以正常接收消息。在 java 控制台中,无论我放入应用程序并发送什么,我都会打印出来。但是第二个Toast是空的(没有传递来自服务器的消息)。我在 LogCat 中得到 SocketException (closed):

01-29 06:38:36.039: W/System.err(11547): java.net.SocketException: Socket is closed
01-29 06:38:36.059: W/System.err(11547):  at java.net.PlainSocketImpl.checkNotClosed(PlainSocketImpl.java:134)
01-29 06:38:36.059: W/System.err(11547):  at java.net.PlainSocketImpl.getInputStream(PlainSocketImpl.java:216)
01-29 06:38:36.059: W/System.err(11547):  at java.net.Socket.getInputStream(Socket.java:343)
01-29 06:38:36.059: W/System.err(11547):  at com.example.simpleclient.ConnectClient.doInBackground(ConnectClient.java:63)
01-29 06:38:36.059: W/System.err(11547):  at com.example.simpleclient.ConnectClient.doInBackground(ConnectClient.java:1)
01-29 06:38:36.059: W/System.err(11547):  at android.os.AsyncTask$2.call(AsyncTask.java:287)
01-29 06:38:36.059: W/System.err(11547):  at java.util.concurrent.FutureTask.run(FutureTask.java:234)
01-29 06:38:36.059: W/System.err(11547):  at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:230)
01-29 06:38:36.059: W/System.err(11547):  at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1080)
01-29 06:38:36.059: W/System.err(11547):  at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:573)
01-29 06:38:36.059: W/System.err(11547):  at java.lang.Thread.run(Thread.java:841)

如果我离开 socket = new Socket(IP, PORT);//second connection to server 代码行中,没有任何错误消息,但消息只传递给服务器一次。第二,第三,结束等等不通过(控制台中没有显示)。尽管如果我让控制台保持运行并关闭应用程序,则会出现另一个空值。

在任何情况下,第二个 Toast(在客户端)要么为空(来自服务器的消息未传递)要么根本不显示(message = bufferedReader.readLine();阻止进一步执行)。例如,如果我注释掉 message = bufferedReader.readLine(); 行,那么我会在第二个 Toast 中得到“after isr”。或者,在第二个 socket = new Socket(IP, PORT); 的情况下//第二个连接 存在,第二个 Toast 根本不显示。

我错过了什么。如何将消息从服​​务器发送回客户端?

最佳答案

客户端和服务器的指令必须是对称的。

如果客户端写入,服务器必须读取,反之亦然

如果客户端打开一个inputStream,服务器必须打开一个outputStream。

现在在第一个连接中你只打开 outputStream 但在服务器中你同时拥有它们。

您还打开了两个在服务器中作为一个连接处理的连接(客户端),因此第一个打印操作工作正常,因为服务器上有读取操作,但其他人无法工作,因为您创建了另一个连接,服务器无法处理,因为:

1)服务器不是多线程的

2)服务器必须在第一个连接上工作

关于java - 通过 tcp 套接字从 java 服务器向 Android 客户端返回消息,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28247364/

相关文章:

java - docker-compose java应用程序连接到mongodb

php - Composer :在phpseclib SSH2中找不到命令

centos - 使用 php-cpp 创建 php 扩展

c# - 更高效的数据包序列化

sockets - 通过网络传输数据时,先将字符串转为字节,再将字节转为字符串,是不是更好,为什么?

java - 将单个构建作业的 .m2 设置到不同的位置

java - AWS 加密 SDK 使用数据 key 加密/解密

java - 如何在android中使用GSON解析JSON

android - 使用电话号码注册

design-patterns - 有没有任何设计模式对使用套接字实现异步通信有帮助?