使用Java / Android中的套接字多次发送二进制数据

时间:2015-03-21 04:00:09

标签: java android python scala sockets

我需要在Android设备上使用Java套接字多次发送二进制数据。 这是一个导出run()和send()方法的简单对象。

public class GpcSocket {

    private Socket socket;

    private static final int SERVERPORT = 9999;
    private static final String SERVER_IP = "10.0.1.4";

    public void run() {
        new Thread(new ClientThread()).start();
    }

    public int send(byte[] str) {
        try {
            final BufferedOutputStream outStream = new BufferedOutputStream(socket.getOutputStream());
            outStream.write(str);
            outStream.flush();
            outStream.close();
        } catch (UnknownHostException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return str.length;
    }

    class ClientThread implements Runnable {
        @Override
        public void run() {
            try {
                InetAddress serverAddr = InetAddress.getByName(SERVER_IP);
                socket = new Socket(serverAddr, SERVERPORT);
            } catch (UnknownHostException e1) {
                e1.printStackTrace();
            } catch (IOException e1) {
                e1.printStackTrace();
            }
        }
    }
}

对于Android编程,我使用Scaloid,这是多次发送二进制数据的代码。 count是从参数中提供的。 result是字节[数组]类型数据,gpc方法onCreate()初始化为gpc = new GpcSocket()

gpc.run()
for (i <- 1 to count) {
  val length = gpc.send(result)
  toast(s"Sent: $i $length")
}

问题在于,即使我尝试多次发送数据,接收器也只接收一个数据包。这是我发送信息5次时在服务器(接收方)端显示的内容:

55:2015-03-21 03:46:51 86: <RECEIVED DATA>
10.0.1.27 wrote:
56:2015-03-21 03:46:51 0:
10.0.1.27 wrote:
57:2015-03-21 03:46:51 0:
10.0.1.27 wrote:
58:2015-03-21 03:46:51 0:
10.0.1.27 wrote:
59:2015-03-21 03:46:51 0:

我的问题是

  • 这是为什么?为什么接收器只接收一次?发件人显示它发送信息五次。
  • 我是否需要在多次发送之前使用run()一次?或者,每当我使用send()时,我是否必须调用run()?

这是Python中的服务器端代码:

import SocketServer
from time import gmtime, strftime

count = 1

class MyTCPHandler(SocketServer.BaseRequestHandler):
    """
    The RequestHandler class for our server.

    It is instantiated once per connection to the server, and must
    override the handle() method to implement communication to the
    client.
    """

    def handle(self):
        # self.request is the TCP socket connected to the client
        self.data = self.request.recv(1024).strip()
        print "{} wrote:".format(self.client_address[0])
        global count
        count += 1
        print "%d:%s %d:%s" % (count, strftime("%Y-%m-%d %H:%M:%S", gmtime()), len(self.data), self.data)
        # just send back the same data, but upper-cased
        self.request.sendall(self.data.upper())

if __name__ == "__main__":
    HOST, PORT = "10.0.1.4", 9999

    # Create the server, binding to localhost on port 9999
    server = SocketServer.TCPServer((HOST, PORT), MyTCPHandler)

    # Activate the server; this will keep running until you
    # interrupt the program with Ctrl-C
    server.serve_forever()

1 个答案:

答案 0 :(得分:0)

从这篇文章:Sending ByteArray using Java Sockets (Android programming)How to fix java.net.SocketException: Broken pipe?,似乎send()方法是错误的。

使用完毕后关闭资源

我将send()修改为简单的创建套接字并在我使用它时将其关闭。

public int send(byte[] str) throws IOException {
    InetAddress serverAddr = InetAddress.getByName(SERVER_IP);
    socket = new Socket(serverAddr, SERVERPORT);
    out = socket.getOutputStream();
    out.write(str);
    out.flush();
    out.close();
    socket.close();
    return str.length;
}

调用者代码中的修改

调用调用它的scala代码以调用send()方法。

// gpc.run()
for (i <- 1 to count) {
  ...
  val length = gpc.send(result)
  println(s"Sent: $i $length")
}

... in initializer
gpc = new GpcSocket()

更改Android中的策略模式

此帖子的提示:android.os.NetworkOnMainThreadException at android.os.StrictMode$AndroidBlockGuardPolicy.onNetwork(StrictMode.java:1145),应为Android设备添加此代码,以防止出现E/AndroidRuntime﹕ FATAL EXCEPTION: main Process: scaloid.example, PID: 1791 android.os.NetworkOnMainThreadException错误。

val policy = new StrictMode.ThreadPolicy.Builder().permitAll().build();
StrictMode.setThreadPolicy(policy);