当wifi断开连接时,Android套接字仍然可以毫无例外地写入

时间:2013-01-07 07:54:34

标签: android sockets wifi

我写了一个Android套接字演示,只是将用户的输入发布到服务器。 我通过wifi连接从android客户端建立一个套接字,一切顺利,服务器可以接收从Android客户端发送的消息。问题是,然后我关闭手机的WIFI,但是套接字可以毫无例外地写入。

Android客户端代码:

public class MyActivity extends Activity {
    private SocketHandlerThread thread;

    /**
     * Called when the activity is first created.
     */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        thread = new SocketHandlerThread("ScoketTest");
        thread.start();
    }

    class SocketHandlerThread extends HandlerThread {
        private Socket socket;
        private Handler handler;

        public SocketHandlerThread(String name) {
            super(name);
        }

        public SocketHandlerThread(String name, int priority) {
            super(name, priority);
        }

        @Override
        public void run() {
            try {
                socket = new Socket("192.168.60.184", 1990);
            } catch (IOException e) {
                Log.e("SocketTest", e.getLocalizedMessage(), e);
            }
            super.run();
        }

        Handler getHandler() {
            if (handler == null) {
                handler = new Handler(getLooper());
            }
            return handler;
        }

        void send(final String text) {
            Runnable runnable = new Runnable() {
                public void run() {
                    Log.e("SocketTest", "Start send text: " + text);
                    try {
                        socket.getOutputStream().write((text + "\n").getBytes());
                        socket.getOutputStream().flush();
                    } catch (Exception e) {
                        Log.e("SocketTest", e.getLocalizedMessage(), e);
                    }
                    Log.e("SocketTest", "Text has been send:" + text);
                }
            };
            getHandler().post(runnable);
        }

        @Override
        protected void onLooperPrepared() {
            runOnUiThread(new Runnable() {
                @Override
                public void run() {
                    findViewById(R.id.button).setEnabled(true);
                }
            });
        }
    }

    public void send(View view) {
        String text = ((EditText) findViewById(R.id.text)).getText().toString();
        thread.send(text);
    }
}

服务器代码:

public class SocketTestServer {
    ServerSocket serverSocket;

    SocketTestServer() throws IOException {
        serverSocket = new ServerSocket(1990);
    }

    void start() throws IOException {
        Socket clientSocket = serverSocket.accept();
        clientSocket.getInputStream();
        PrintWriter out = new PrintWriter(System.out, true);
        BufferedReader in =
                new BufferedReader(new InputStreamReader(clientSocket.getInputStream()));
        String inputLine;
        while ((inputLine = in.readLine()) != null) {
            out.println(inputLine);
        }
    }

    public static void main(String[] args) throws IOException {
        SocketTestServer server = new SocketTestServer();
        server.start();
    }
}

我尝试了几款手机。在Galaxy Nexus(4.2.1)上,如预期的那样抛出异常,但在某些MOTO或HTC手机上,套接字仍然可以毫无例外地写入,这意味着我可能会丢失一些我认为已成功接收的消息。 我怎么能知道套接字连接在任何类型的手机上都被破坏了?

任何建议都将受到赞赏。

P.S 我知道连接更改广播,但在接收广播之前,客户端可能已经通过损坏的套接字写了一些消息。

虽然在应用程序协议上添加接收检查可以解决消息丢失问题,但我更愿意保证TCP协议承诺的传输层的可靠性。

1 个答案:

答案 0 :(得分:0)

这是TCP堆栈(OS级别)的问题,并且是难以解决的问题。 如果服务器是丢失连接的服务器(尝试在打开套接字后重新启动服务器而没有客户端会注意到),也会发生这种情况...下次通过套接字发送数据包时,您将遇到“管道错误”。

所以,有两种情况:

1-客户端发送信息时损坏的管道。

在这种情况下,您应该捕获IOException并重试打开连接(您必须定义重试策略)。

当服务器失去连接时,

2-断管

当服务器失去连接时,客户端不会注意到这一点,在这种情况下,您应该定期向服务器发送数据包(轮询技术)并尝试重新连接,以防连接丢失。 如果您从服务器更新收到,如果您的流量始终是客户端 - > gt;服务器仅适用方案1,则需要这样做。