首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Android UDP通信

Android UDP通信
EN

Stack Overflow用户
提问于 2011-05-05 22:28:01
回答 3查看 17.9K关注 0票数 7

我在这个网站上读过很多关于如何在Android中接收UDP数据包的文章。然而,所有这些都不适合我!

一些基本知识:

我正在测试我的HTC不可思议(Android2.2)运行在3G上(而不是wifi或其他任何东西)。这里不涉及仿真器。

我的代码很简单:

  1. 我的服务器(运行在我的PC上)正在监听8752端口上的UDP流量。
  2. 我的
  3. 应用程序在一个随机端口上打开一个DatagramSocket,并用这个端口向我的服务器发送一个数据包。
  4. 然后保存这些信息(从接收到的数据包和在数据包中找到的端口中找到的InetAddress )。
  5. 我尝试从我的服务器(再次在我的PC上)发送一个UDP数据包到我的Android应用程序(运行在我的手机上),但它不工作。

代码语言:javascript
复制
//Server code to initialize the UDP socket (snippet)
public void init() {
    datagram_server_socket = new DatagramSocket(port,local_addr);
    datagram_server_socket.setSoTimeout(1000);
}

//向服务器发送数据包的ANDROID应用程序上的代码片段

代码语言:javascript
复制
public void connect() {
    Random r = new Random(System.currentTimeMillis());
    int udp_port = 0;
    while(true){
        try {
            udp_port = r.nextInt(1000)+8000;
            udp_port = 8000;
            comm_skt = new DatagramSocket(udp_port);
            Log.i("ServerWrapper", "UDP Listening on port: " + udp_port);
            break;
        } catch(SocketException e) {
            Log.e("ServerWrapper", "Could not bind to port " + udp_port);
        }
    }
    byte[] sdata = new byte[4+tid.length];
    i = 0;
    sdata[i++] = (byte)(0XFF&(udp_port>>24));
    sdata[i++] = (byte)(0XFF&(udp_port>>16));
    sdata[i++] = (byte)(0XFF&(udp_port>>8));
    sdata[i++] = (byte)(0XFF&(udp_port));
    for(byte b: tid){
        sdata[i++] = b;
    }
    DatagramPacket pkt = new DatagramPacket(sdata, sdata.length, 
                                InetAddress.getByName(hostname), port);
    comm_skt.send(pkt);
}
代码语言:javascript
复制
//Server's UDP socket listening code
public void serverUDPListener() {
    try {
        datagram_server_socket.receive(rpkt);
        int port = 0;
        byte[] rdata = rpkt.getData();
        port += rdata[0]<<24;
        port += rdata[1]<<16;
        port += rdata[2]<<8;
        port += (0XFF)&rdata[3];
        byte[] tid = new byte[rdata.length];
        for(int i = 4; i < rdata.length && rdata[i] > 0; i++) {
            tid[i-4] = rdata[i];
        }
        String thread_id = new String(tid).trim();
        for(int i = 0; i < threads.size(); i++) {
        ClientThread t = threads.get(i);
        if(t.getThreadId().compareTo(thread_id) == 0) {
            t.setCommSocket(rpkt, port);
        } else {
            System.err.println("THREAD ID " + thread_id + " COULD NOT BE FOUND");
        }
        }
    } catch (IOException e) {
        if(!(e instanceof SocketException) && !(e instanceof SocketTimeoutException))
        log.warning("Error while listening for an UDP Packet.");
    }
}
代码语言:javascript
复制
//Corresponds to the setCommSocket call above to save the IP and Port of the incoming UDP packet on the server-end
public void setCommSocket(DatagramPacket pkt, int port) {
    comm_ip = pkt.getAddress();
    comm_port = pkt.getPort(); //Try the port from the packet?
}
代码语言:javascript
复制
//Sends an UDP packet from the SERVER to the ANDROID APP
public void sendIdle() {
    if(comm_ip != null) {
        System.err.println("Sent IDLE Packet (" + comm_ip.getHostAddress() + ":" + comm_port + ")");
        DatagramPacket spkt = new DatagramPacket(new byte[]{1, ProtocolWrapper.IDLE}, 2, comm_ip, comm_port);
        DatagramSocket skt;
        try {
            skt = new DatagramSocket();
            skt.send(spkt);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

现在,我已经将我的应用程序使用的端口硬编码为8000。然而,奇怪的是,每次我测试程序(并查看保存在服务器上的IP/端口)时,数据包的端口总是33081。在我的Android应用程序中,我有一个线程不断地侦听UDP通信量,但是代码从不执行传递的“接收(数据包)”部分:

代码语言:javascript
复制
public void AndroidUDPListener() {
    while(true) {
        synchronized(stop) {
        if(stop) return;
        }
        byte[] recieve_data = new byte[64];
        DatagramPacket rpkt = new DatagramPacket(recieve_data, recieve_data.length);
        try {
        if(comm_skt == null) 
                continue;
        comm_skt.receive(rpkt);
        byte[] data = rpkt.getData();
        switch(data[1]) {
            case IDLE:
            if(ocl != null) ocl.onCompletion(null);
            break;
            case KEEP_ALIVE:
            break;
        }
        } catch (Exception e) {
        if(!(e instanceof SocketException) && !(e instanceof SocketTimeoutException))
                Log.w("ServerWrapper", "Error while listening for an UDP Packet.");
        }
    }
}

有人在我的代码中看到问题了吗?或者我需要先在我的应用程序上设置一些权限/设置?我已经启用了网络通讯。

示例输出(使用来自数据包getPort()的端口):

安卓应用程序-现在监听端口8000上的UDP流量

Android应用程序-向服务器发送数据包

服务器-从XXXXXX:33081收到的数据包

服务器-将空闲数据包发送到XXXXXX:33081

示例输出(使用来自数据包数据的端口):

安卓应用程序-现在监听端口8000上的UDP流量

Android应用程序-向服务器发送数据包

服务器-从XXXXXX:8000收到的数据包

服务器-将空闲数据包发送到XXXXXX:8000

Android应用程序从未从任何一个端口接收到任何UDP流量。

EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2012-10-03 04:19:07

很抱歉没能尽快更新。问题的解决办法如下:

我需要将DatagramSocket存储到每个线程。侦听套接字也应该是用于在服务器和客户端之间继续通信的套接字。以下是更新后的代码。

线程上的新套接字注册代码:

代码语言:javascript
复制
public void setCommSocket(DatagramPacket pkt, int port, DatagramSocket skt)
{
  comm_ip = pkt.getAddress();
  comm_port = pkt.getPort();
  synchronized(comm_pkt) {
    comm_pkt = pkt;
  }
  comm_skt = skt;
}

新服务器侦听代码:

代码语言:javascript
复制
public void UDPListen() {
        while(true) {
            synchronized(stop) {
                if(stop)
                    break;
            }

            byte[] recieve_data = new byte[64];
            DatagramPacket rpkt = new DatagramPacket(recieve_data, recieve_data.length);
            try {
                datagram_server_socket.receive(rpkt);
                int port = 0;
                byte[] rdata = rpkt.getData();
                port += rdata[0]<<24;
                port += rdata[1]<<16;
                port += rdata[2]<<8;
                port += (0XFF)&rdata[3];
                byte[] tid = new byte[rdata.length];
                for(int i = 4; i < rdata.length && rdata[i] > 0; i++)
                {
                    tid[i-4] = rdata[i];
                }
                String thread_id = new String(tid).trim();
                for(int i = 0; i < threads.size(); i++) {
                    ClientThread t = threads.get(i);
                    if(t.getThreadId().compareTo(thread_id) == 0)
                    {
                        t.setCommSocket(rpkt, port, datagram_server_socket);
                    } else {
                        System.err.println("THREAD ID " + thread_id + " COULD NOT BE FOUND");
                    }
                }
            } catch (IOException e) {
                if(!(e instanceof SocketException) && !(e instanceof SocketTimeoutException))
                    log.warning("Error while listening for an UDP Packet.");
            } finally {
                for(int i = 0; i < threads.size(); i++) {
                    ClientThread t = threads.get(i);
                    t.sendKeepAlive();
                }
            }
        }
    }

服务器/线程的结构有了一些更新,我将省略这些更新。这里重要的部分是重新使用数据包接收的套接字将数据发送回客户端。此外,实际的数据包被重新用于将数据发回:

代码语言:javascript
复制
public void sendIdle() {
        if(comm_ip != null) {
            synchronized(comm_pkt) {
                try {
                    comm_pkt.setData(new byte[]{1, ProtocolWrapper.IDLE});
                    comm_skt.send(comm_pkt);
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }

    }

下面是我的包装类的相关部分,它显示了每个线程所包含的内容:

代码语言:javascript
复制
public class PeerWrapper {

    private InetAddress ipaddress;
    private Integer port;
    private Socket client_socket;
    private InetAddress comm_ip;
    private DatagramSocket comm_skt;
    private DatagramPacket comm_pkt;
    private int comm_port;
    private byte status;
票数 4
EN

Stack Overflow用户

发布于 2012-04-01 01:16:33

我也有过类似的问题。Android上有两个套接字(发送/侦听),在PC服务器上又有两个套接字(发送/侦听)。电话会将PC的已知监听套接字与电话的未知监听套接字的地址进行切换,这样PC就可以回复。我所做的任何事情似乎都没有得到监听套接字的地址,因为套接字永远不会接收到任何东西。

这解决了我的问题:Android: java.net.DatagramSocket.bind: Invalid Argument Exception。使用一个通道创建套接字,然后绑定到null上。现在我可以使用手机上的发送套接字向PC发送包含侦听套接字端口(I是相同的)的数据包,用.getLocalPort()获得,PC读取byte[],获取端口,并将数据包发送回电话侦听端口。

票数 2
EN

Stack Overflow用户

发布于 2012-06-20 10:53:50

android有入站防火墙

你必须先用udop孔打孔和计时器打孔。

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/5904820

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档