在Android中使用NAT打洞进行P2P消息发送

3
我试图使用Kryonet(使用socket通信)从一部手机发送消息到另一部手机,并且我打算通过存储客户端的公共地址来使用NAT穿透。
以下代码被使用:
public class MainActivity extends Activity implements View.OnClickListener
{
  /**
   * Called when the activity is first created.
   */
  @Override
  public void onCreate(Bundle savedInstanceState)
  {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    Button button = ((Button) findViewById(R.id.btnTest));
    button.setOnClickListener(this);
    TextView textViewOwn = ((TextView) findViewById(R.id.lblOwnIP));
    textViewOwn.setText(getLocalIpAddress()+"\n"+GetGateway());
    try
    {
      Server server = new Server();
      server.start();
      server.bind(54555, 54777);
      Log.i("DEBUG", "Server is listening");
      final TextView textView = ((TextView) findViewById(R.id.txtMessage));
      server.addListener(new Listener()
      {
        public void received(Connection connection, Object object)
        {
          if (object instanceof String)
          {
            String request = (String) object;
            Log.i("DEBUG", request);
            ShowMessage(request, textView);
            String response = "Thanks";
            connection.sendTCP(response);
          }
        }
      });

    }
    catch (Exception e)
    {
      e.printStackTrace();
    }
  }

  public static String getLocalIpAddress() {
    try {
      for (Enumeration<NetworkInterface> en = NetworkInterface.getNetworkInterfaces(); en.hasMoreElements();) {
        NetworkInterface intf = en.nextElement();
        for (Enumeration<InetAddress> enumIpAddr = intf.getInetAddresses(); enumIpAddr.hasMoreElements();) {
          InetAddress inetAddress = enumIpAddr.nextElement();
          if (!inetAddress.isLoopbackAddress() && inetAddress instanceof Inet4Address) {
            return inetAddress.getHostAddress();
          }
        }
      }
    } catch (SocketException ex) {
      ex.printStackTrace();
    }
    return null;
  }

  private String GetGateway()
  {
    DhcpInfo d;
    WifiManager wifii;
    wifii= (WifiManager) getSystemService(Context.WIFI_SERVICE);
    d=wifii.getDhcpInfo();
    int gatewayip = d.gateway;
    String mask = Formatter.formatIpAddress(gatewayip);
    return mask;
  }

  private void ShowMessage(final String request, final TextView textView)
  {
    runOnUiThread(new Runnable()
    {
      @Override
      public void run()
      {
        textView.setText(request);
      }
    });
  }

  @Override
  public void onClick(View view)
  {
    final String ip = ((EditText) findViewById(R.id.txtIP)).getText().toString();
    if (TextUtils.isEmpty(ip))
    {
      Toast.makeText(this, "No IP address", Toast.LENGTH_SHORT).show();
      return;
    }
    Runnable runnable = new Runnable()
    {
      public void run()
      {
        try
        {
          Client client = new Client();
          client.start();

          client.connect(5000, ip, 54555, 54777);
          Log.i("DEBUG", "Client is sending: "+ip);
          runOnUiThread(new Runnable()
          {
            @Override
            public void run()
            {
              Toast.makeText(MainActivity.this, "Client is sending: "+ip, Toast.LENGTH_SHORT).show();
            }
          });

          String request = "Here is the request";
          client.sendTCP(request);
        }
        catch (final Exception e)
        {
          e.printStackTrace();
          runOnUiThread(new Runnable()
          {
            @Override
            public void run()
            {
              Toast.makeText(MainActivity.this, "Client has errors: "+e.toString(), Toast.LENGTH_SHORT).show();

            }
          });
        }
      }
    };
       Thread thread=new Thread(runnable);
    thread.start();
  }
}

如果手机在同一个WIFI上,它可以正常工作。如果其中一个手机使用3G网络,则会获得外部IP地址(例如:100.65.96..),并获得本地地址作为网关。由于了解外部IP是NAT PT的关键,我认为这已足够接收消息,但我却收到了超时的消息。

我还需要实现什么来让使用3G网络的手机接收消息?

1个回答

1
您可以通过使用中介服务器来建立手机之间的连接。
1. 创建一个能够接受UDP套接字并存储连接到它的手机的公共地址的服务器程序。 2. 使用“手机A”连接到该服务器以注册其公共地址。 3. 使用“手机B”连接到服务器以请求“手机A”的公共地址。 4. 一旦获得该地址,您可以通过使用其公共地址从“手机A”到“手机B”创建一个“直接套接字”。NAT将完成其余工作,路由器将向手机的私有地址发送数据包。

如果你想使用TCP而不是UDP怎么办?如果不可能的话...我想我必须在UDP中实现TCP所提供的一切。 - ArtOfWarfare
我认为运营商会阻止任何直接进入A的连接,除非A之前已经直接联系过。 - Allahjane

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接