Java Android-UDP通信

Java Android-UDP通信,java,android,udp,Java,Android,Udp,我在这个网站上读了很多关于如何在Android中接收UDP数据包的帖子。然而,这些都不适合我 一些基本知识: 我正在测试运行在3G(不是wifi或其他任何东西)上的HTC Inclusible(Android 2.2)。这里不涉及模拟器 我的代码很简单: 我的服务器(在我的电脑上运行)正在监听端口8752上的UDP通信 我的Android应用程序在一个随机端口上打开一个DatagramSocket,并用这个端口向我的服务器发送一个数据包 然后保存该信息(接收到的数据包的InetAddress和在

我在这个网站上读了很多关于如何在Android中接收UDP数据包的帖子。然而,这些都不适合我

一些基本知识:

我正在测试运行在3G(不是wifi或其他任何东西)上的HTC Inclusible(Android 2.2)。这里不涉及模拟器

我的代码很简单:

  • 我的服务器(在我的电脑上运行)正在监听端口8752上的UDP通信
  • 我的Android应用程序在一个随机端口上打开一个DatagramSocket,并用这个端口向我的服务器发送一个数据包
  • 然后保存该信息(接收到的数据包的InetAddress和在数据包中找到的端口)
  • 我尝试从我的服务器(同样,在我的PC上)向我的Android应用程序(在我的手机上运行)发送UDP数据包,但它不起作用
  • //初始化UDP套接字的服务器代码(代码段)
    公共void init(){
    datagram_server_socket=新DatagramSocket(端口,本地地址);
    数据报服务器套接字设置超时(1000);
    }
    
    //ANDROID应用程序上向服务器发送数据包的代码片段

    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);
    }
    
    //服务器的UDP套接字侦听代码
    public void serverUDPListener(){
    试一试{
    数据报服务器套接字接收(rpkt);
    int端口=0;
    字节[]rdata=rpkt.getData();
    
    port+=rdata[0]我也遇到了类似的问题。Android上有两个套接字(发送/侦听),PC服务器上也有两个套接字(发送/侦听)。手机会用手机未知的监听插座的地址ping电脑的已知监听插座,这样电脑就可以回复。我所做的任何事情似乎都无法获取监听插座的地址,因为该插座永远不会接收任何内容


    这解决了我的问题:。使用通道创建套接字,然后绑定为null。现在,我可以使用手机上的发送套接字将包含侦听套接字端口(IP相同)的数据包发送到PC,该数据包是通过
    获得的。getLocalPort()
    PC读取字节[],获取端口,并将数据包发送回电话侦听端口。

    android具有入站防火墙


    您必须使用带计时器的第一个udop hole punch同一袜子对象。抱歉,没有提前更新此对象。问题已修复,如下所示:

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

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

    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;
    }
    
    新服务器侦听代码:

    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();
                    }
                }
            }
        }
    
    下面是我的包装器类的相关部分,显示了每个线程所持有的内容:

    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;
    

    你能在两台台式机/笔记本电脑(没有你的android手机)上运行吗?如果你能消除手机的问题,那么我们可以认为这是你的代码的问题。我可以把我的手机放在我的WiFi网络上,让UDP包以这种方式工作(使用直接IP到我的机器作为主机名,使用外部DNS名称不起作用)。我还让我的朋友测试了一个我编写的程序,在他家模拟Android的应用程序网络调用(使用外部DNS名称),它工作得很好(在做了一些端口转发并强制使用非随机UDP端口后)。我希望这能澄清我的代码问题。您好,谢谢您的回答。我不确定它是否能解决我的问题,但我在一年多前完成了这个项目(并解决了我的问题),没有时间测试它是否有效。
    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;