【问题标题】:W/System.err: java.net.SocketException: Socket closedW/System.err: java.net.SocketException: 套接字关闭
【发布时间】:2016-07-03 03:29:53
【问题描述】:

我有问题。我关闭此应用程序收到错误。尽管尚未访问套接字服务器。 怎么解决?套接字服务器是片段。可以影响这个吗?

    W/System.err: java.net.SocketException: Socket closed
W/System.err:     at java.net.PlainSocketImpl.socketAccept(Native Method)
W/System.err:     at java.net.AbstractPlainSocketImpl.accept(AbstractPlainSocketImpl.java:389)
W/System.err:     at java.net.ServerSocket.implAccept(ServerSocket.java:531)
W/System.err:     at java.net.ServerSocket.accept(ServerSocket.java:499)
W/System.err:     at ro.vrt.videoplayerstreaming.Torrent_remote$SocketServerThread.run(Torrent_remote.java:92)
W/System.err:     at java.lang.Thread.run(Thread.java:761)

              --------- beginning of crash
E/AndroidRuntime: FATAL EXCEPTION: Thread-9
                  Process: ro.vrt.videoplayerstreaming, PID: 2652
                  java.lang.NullPointerException: Attempt to invoke virtual method 'void android.support.v4.app.FragmentActivity.runOnUiThread(java.lang.Runnable)' on a null object reference
                      at ro.vrt.videoplayerstreaming.Torrent_remote$SocketServerThread.run(Torrent_remote.java:134)
                      at java.lang.Thread.run(Thread.java:761)
D/FA: Application backgrounded. Logging engagement
D/FA: Logging event (FE): _e, Bundle[{_o=auto, _et=560890}]
V/FA: Using local app measurement service
V/FA: Local AppMeasurementService is starting up
V/FA: Bound to IMeasurementService interface
V/FA: Connected to service
V/FA: Processing queued up service tasks: 1
V/FA: Logging event: origin=auto,name=_e,params=Bundle[{_o=auto, _et=560890}]
D/FA: Unable to get advertising id: com.google.android.gms.common.GooglePlayServicesNotAvailableException: com.google.android.gms.measurement.internal.zzt.zzly(Unknown Source)
V/FA: Saving event, name, data size: _e, 30
V/FA: Event recorded: Event{appId='ro.vrt.videoplayerstreaming', name='_e', params=Bundle[{_o=auto, _et=560890}]}
V/FA: Upload scheduled in approximately ms: 3095074
V/FA: Background event processing time, ms: 20
V/FA: Inactivity, disconnecting from AppMeasurementService
V/FA: onUnbind called for intent. action: com.google.android.gms.measurement.START
V/FA: Local AppMeasurementService is shutting down

代码:

    mport android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;

import android.app.Activity;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.EditText;
import android.widget.TextView;

import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.net.InetAddress;
import java.net.NetworkInterface;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketException;
import java.util.Enumeration;

public class Torrent_remote extends Fragment {

    TextView info, infoip, msg;
    String message = "";
    ServerSocket serverSocket;


    @Nullable
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        View v = inflater.inflate(R.layout.activity_torrent_remote, container, false);

        info = (TextView) v.findViewById(R.id.info);
        infoip = (TextView) v.findViewById(R.id.infoip);
        msg = (TextView) v.findViewById(R.id.msg);

        infoip.setText(getIpAddress());

        Thread socketServerThread = new Thread(new SocketServerThread());
        socketServerThread.start();


        return v;
    }

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

        if (serverSocket != null) {
            try {
                serverSocket.close();
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }

    }

    private class SocketServerThread extends Thread {

        static final int SocketServerPORT = 8080;
        int count = 0;

        @Override
        public void run() {
            Socket socket = null;
            DataInputStream dataInputStream = null;
            DataOutputStream dataOutputStream = null;

            try {
                serverSocket = new ServerSocket(SocketServerPORT);
                getActivity().runOnUiThread(new Runnable() {

                    @Override
                    public void run() {
                        info.setText("I'm waiting here: "
                                + serverSocket.getLocalPort());
                    }
                });

                while (true) {
                    socket = serverSocket.accept();
                    dataInputStream = new DataInputStream(
                            socket.getInputStream());
                    dataOutputStream = new DataOutputStream(
                            socket.getOutputStream());

                    String messageFromClient = "";

                    //If no message sent from client, this code will block the program
                    messageFromClient = dataInputStream.readUTF();

                    count++;
                    message += "#" + count + " from " + socket.getInetAddress()
                            + ":" + socket.getPort() + "\n"
                            + "Msg from client: " + messageFromClient + "\n";

                    getActivity().runOnUiThread(new Runnable() {

                        @Override
                        public void run() {
                            msg.setText(message);
                        }
                    });

                    String msgReply = "Hello from VRT Player Android Tv, msg nr #" + count;
                    dataOutputStream.writeUTF(msgReply);

                    String url = messageFromClient;
                    //Put the value
                    Torrent_fragment ldf = new Torrent_fragment();
                    Bundle args = new Bundle();
                    args.putString("url", url);
                    ldf.setArguments(args);
                    getFragmentManager().beginTransaction().replace(R.id.frame, ldf).commit();



                }
            } catch (IOException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
                final String errMsg = e.toString();
                getActivity().runOnUiThread(new Runnable() {

                    @Override
                    public void run() {
                        msg.setText(errMsg);
                    }
                });

            } finally {
                if (socket != null) {
                    try {
                        socket.close();
                    } catch (IOException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }

                if (dataInputStream != null) {
                    try {
                        dataInputStream.close();
                    } catch (IOException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }

                if (dataOutputStream != null) {
                    try {
                        dataOutputStream.close();
                    } catch (IOException e) {
                        // TODO Auto-generated catch block
                        e.printStackTrace();
                    }
                }
            }
        }

    }

    private String getIpAddress() {
        String ip = "";
        try {
            Enumeration<NetworkInterface> enumNetworkInterfaces = NetworkInterface
                    .getNetworkInterfaces();
            while (enumNetworkInterfaces.hasMoreElements()) {
                NetworkInterface networkInterface = enumNetworkInterfaces
                        .nextElement();
                Enumeration<InetAddress> enumInetAddress = networkInterface
                        .getInetAddresses();
                while (enumInetAddress.hasMoreElements()) {
                    InetAddress inetAddress = enumInetAddress.nextElement();

                    if (inetAddress.isSiteLocalAddress()) {
                        ip += "SiteLocalAddress: "
                                + inetAddress.getHostAddress() + "\n";
                    }

                }

            }

        } catch (SocketException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
            ip += "Something Wrong! " + e.toString() + "\n";
        }

        return ip;
    }
}

并非总是会出现此错误。 谢谢

【问题讨论】:

  • 真的有问题吗?这一切意味着您在另一个线程中的accept() 中被阻塞时关闭了他的套接字。这是一件可以做的事情,结果就是这个异常。忽略它。 NB 'socket server has not been access' 和它没有任何关系。

标签: java android sockets


【解决方案1】:

您应该检查套接字是否已经关闭(在关闭它之前使用serverSocket.isClosed()。因此,将您的onDestroy 代码更改为:

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

    if (serverSocket != null && !serverSocket.isClosed()) {
        try {
            serverSocket.close();
        } catch (IOException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

}

更新

看来问题出在这个Thread 上:

SocketServerThread

该线程正在片段内运行,因此请确保在调用 socket.close(); 之前关闭此 Thread,因为您在此 Thread 中使用 socket.accept(),因此在您关闭 @987654331 时会导致错误@inside onDestory 方法,然后它尝试从 Threadsocket.accept(),因为它仍在运行。

【讨论】:

  • 那么最好的方法是在关闭应用程序之前,onback 的活动从片段发送 socket.close ?
  • 查看如何安全地停止线程:*.com/questions/8505707/…
【解决方案2】:

getActivity() 返回 null,因为在执行 onActivityCreated() 回调时对活动对象的引用可用,而不是在 onCreateView() 上。

我建议你在onActivityCreated() 回调中启动线程。

【讨论】: