连接到远程 tcp 套接字时我的应用程序崩溃

My app crashes when connecting to remote tcp socket

我已经尝试了几个小时让我的应用程序(运行 在 Android 7.1.2 上)连接到我家庭局域网中台式电脑上的服务器套接字。

public class MainActivity extends AppCompatActivity {

    static final String SERVER_IP   = "192.168.XXX.XXX"; // X's are replaced by actual IP
    static final int    PORT        = 30000;

    Socket clientSocket;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        final Button btReceive  = (Button) findViewById(R.id.btReceive);

        final TextView txtReceivedText  = (TextView) findViewById(R.id.txtReceivedText);
        final TextView txtStatus        = (TextView) findViewById(R.id.txtStatus);



        btReceive.setOnClickListener(new View.OnClickListener(){
            @Override
            public void onClick(View v) {

                txtStatus.setText("Received button pressed.");

                try {

                    Socket s = new Socket(SERVER_IP, PORT);


                    BufferedReader input = new BufferedReader(new InputStreamReader(s.getInputStream()));
                    final String st = input.readLine();

                    txtReceivedText.setText(st);

                    s.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }

        });

    }


}

执行时,我在 Android studio 的控制台上得到输出:

...
Waiting for process to come online
Connected to process 16980 on device samsung-gt_i9515-a72266ed
Capturing and displaying logcat messages from application. This behavior can be disabled in the "Logcat output" section of the "Debugger" settings page.
W/art: Before Android 4.1, method int android.support.v7.widget.ListViewCompat.lookForSelectablePosition(int, boolean) would have incorrectly overridden the package-private method in android.widget.ListView
D/AndroidRuntime: Shutting down VM
E/AndroidRuntime: FATAL EXCEPTION: main
                  Process: com.example.jonas.network_connector, PID: 16980
                  android.os.NetworkOnMainThreadException
                      at android.os.StrictMode$AndroidBlockGuardPolicy.onNetwork(StrictMode.java:1303)
                      at java.net.AbstractPlainSocketImpl.doConnect(AbstractPlainSocketImpl.java:333)
                      at java.net.AbstractPlainSocketImpl.connectToAddress(AbstractPlainSocketImpl.java:196)
                      at java.net.AbstractPlainSocketImpl.connect(AbstractPlainSocketImpl.java:178)
                      at java.net.SocksSocketImpl.connect(SocksSocketImpl.java:356)
                      at java.net.Socket.connect(Socket.java:605)
                      at java.net.Socket.connect(Socket.java:554)
                      at java.net.Socket.<init>(Socket.java:431)
                      at java.net.Socket.<init>(Socket.java:210)
                      at com.example.jonas.network_connector.MainActivity.onClick(MainActivity.java:88)
                      at android.view.View.performClick(View.java:5637)
                      at android.view.View$PerformClick.run(View.java:22433)
                      at android.os.Handler.handleCallback(Handler.java:751)
                      at android.os.Handler.dispatchMessage(Handler.java:95)
                      at android.os.Looper.loop(Looper.java:154)
                      at android.app.ActivityThread.main(ActivityThread.java:6186)
                      at java.lang.reflect.Method.invoke(Native Method)
                      at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:889)
                      at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:779)
I/Process: Sending signal. PID: 16980 SIG: 9
Application terminated.

由于我是套接字编程的新手(android 也是),我不知道如何继续查找错误。

我的清单有权限行:

<uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />

android.os.NetworkOnMainThreadException

此异常告诉您您正试图在主线程中调用远程。尝试使用 AsyncTask 调用,应该不会再出现问题。 Android OS 阻止在主线程中执行长 运行 操作,因为它可以锁定 UI.

android.os.NetworkOnMainThreadException

这意味着您的应用程序试图在其主线程上执行网络操作。 运行 您的代码在 AsyncTask.

示例:

 class TCPTask extends AsyncTask<Void, Void, Void>{

        @Override
        protected void onPreExecute() {
            super.onPreExecute();
        }

        @Override
        protected Void doInBackground(Void... params) {
            return null;
        }

        @Override
        protected void onPostExecute(Void aVoid) {
            super.onPostExecute(aVoid);
        }
    }

执行异步任务:

将此代码添加到您的 oncreate() 方法中

new TCPTask().execute();

您需要在工作线程上执行网络工作。

btReceive.setOnClickListener(new View.OnClickListener(){
            @Override
            public void onClick(View v) {
                txtStatus.setText("Received button pressed.");
                new Thread(new Runnable() {
                    @Override
                    public void run() {
                        try {
                            Socket s = new Socket(SERVER_IP, PORT);
                            BufferedReader input = new BufferedReader(new InputStreamReader(s.getInputStream()));
                            final String st = input.readLine();
                            txtReceivedText.post(new Runnable(){

                                @Override
                                public void run() {
                                    txtReceivedText.setText(st);
                                }
                            });

                            s.close();
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                    }
                    }
                }).start();
            });