如何在 Android 上使用 UDP 数据报在两个进程和两个主机之间进行通信
How to communicate between two process and two hosts using UDP datagrams on Android
我的分布式应用程序,有多个主机,需要在同一局域网内找到本地服务器。
我使用 NsdManager 检测本地服务器的成功率有限,这是要走的路,但通常,服务器仍未检测到。我正在寻找更好的解决方案,更可靠的东西。
然后我尝试多播 UDP 数据包,我在同一进程中检测到没有问题。 (我的单元测试同时使用客户端和服务器实例)。那很好用。同样 类 将不允许我的客户端在同一设备上检测到我的服务器 运行(客户端和服务器是单独的进程应用程序)。
然后我改变了使用广播地址的方法,同样,在同一个进程中,单元测试很好,但在两个不同的进程中,找不到服务器。
正如我所说,代码在同一个进程中工作正常,但当 运行 在两个不同的进程中,或在同一 LAN 中的两个不同的客户端/服务器主机上时,它们不起作用。
注意:我的局域网是高端WiFi rooter,没有特殊设置。
有人知道问题出在哪里吗?
请注意,我的清单包括客户端和服务器应用程序的这些设置:
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_MULTICAST_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
请注意,为了让我的客户端检测同一主机上的服务器 运行,我已经在使用 TCP 套接字,但我需要检测局域网中的任何服务器。
@Onik 说的很对,
制作一个最小的、完整的、可验证的例子让我找到了问题。
以下是可能感兴趣的人的代码:
package com.example;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.InterfaceAddress;
import java.net.NetworkInterface;
import java.util.Enumeration;
import java.util.List;
import android.util.Log;
/**
* Minimal, Complete, and Verifiable example for broadcasting UDP sockets, send thread.
*
* Loosely based on:
*
* https://jackiexie.com/2015/07/15/network-discovery-using-udp-broadcast-java/
*/
public class BroadcastSender extends Thread {
public static final String TAG = "BroadcastSender";
public static final int TEST_PORT_NUMBER = 1234;
public static final int TEST_COUNT = 10;
public static final int DELAY_BETWEEN_PACKETS_MS = 500;
public static final int TEST_TIME_MS = TEST_COUNT * DELAY_BETWEEN_PACKETS_MS;
private DatagramSocket mSocket;
public BroadcastSender() {
super( TAG );
}
public static InetAddress getBroadcastAddress() {
try {
Enumeration<NetworkInterface> interfaces = NetworkInterface.getNetworkInterfaces();
while( interfaces.hasMoreElements() ) {
NetworkInterface current = interfaces.nextElement();
try {
if( !current.isUp() )
continue;
if( current.isLoopback() )
continue;
List<InterfaceAddress> addresses = current.getInterfaceAddresses();
if( addresses == null ) {
log("getBroadcastAddress can not get addresses from "+current);
continue;
}
for( InterfaceAddress oneInterfaceAddress : addresses ) {
InetAddress bcast = oneInterfaceAddress.getBroadcast();
if( bcast != null )
return bcast;
}
}
catch( Exception ex) {
log( "getBroadcastAddress exception "+ex+" stack "+Log.getStackTraceString(ex) );
}
}
}
catch( Exception ex) {
log( "getBroadcastAddress exception "+ex+" stack "+Log.getStackTraceString(ex) );
}
return null;
}
@Override
public void run() {
try {
mSocket = new DatagramSocket();
mSocket.setBroadcast(true);
InetAddress interfaceBroadcast = getBroadcastAddress();
InetAddress broadcastAddr = InetAddress.getByName("255.255.255.255");
String payloadStr = "payload pid="+android.os.Process.myPid();
byte payloadBytes[] = payloadStr.getBytes();
for( int packet = 0; packet < TEST_COUNT; packet++) {
// First send to 255.255.255.255
DatagramPacket dp = new DatagramPacket(payloadBytes, payloadBytes.length, broadcastAddr, TEST_PORT_NUMBER);
mSocket.send(dp);
// Then send to whatever address we have for interfaceBroadcast
if( interfaceBroadcast != null ) {
dp = new DatagramPacket(payloadBytes, payloadBytes.length, interfaceBroadcast, TEST_PORT_NUMBER);
mSocket.send(dp);
}
log("sent #"+packet+" with "+payloadBytes.length+" bytes \""+payloadStr+"\"");
synchronized (this) {
try {
wait(DELAY_BETWEEN_PACKETS_MS);
}
catch( InterruptedException ex) {
log("run exeption "+ex.toString());
}
}
}
mSocket.close();
mSocket = null;
}
catch( Exception ex) {
log("run exception "+ex.toString()+" stack "+Log.getStackTraceString(ex));
}
}
private static void log( String message) {
Log.d(TAG, message);
}
/**
* Minimal, Complete, and Verifiable example for broadcasting UDP sockets, receive thread.
* Loosely based on:
*
* https://jackiexie.com/2015/07/15/network-discovery-using-udp-broadcast-java/
*/
public static class BroadcastReceiver extends Thread {
public static final String TAG = "BroadcastReceiver";
public int totalReceived = 0;
public BroadcastReceiver() {
super( TAG );
}
@Override
public void run() {
final String methodName = "run";
try {
byte messageBytes[] = new byte[1500];
DatagramPacket dp = new DatagramPacket(messageBytes, messageBytes.length);
DatagramSocket mSocket = new DatagramSocket(TEST_PORT_NUMBER);
mSocket.setSoTimeout(DELAY_BETWEEN_PACKETS_MS);
mSocket.setBroadcast(true);
while( totalReceived < TEST_COUNT ) {
try {
// Get one UDP packet
mSocket.receive(dp);
if( dp.getLength() > 0 ) {
totalReceived++;
String payloadStr = new String(dp.getData());
log( "received #"+totalReceived+" : \""+payloadStr+"\"");
}
}
catch( Exception ex) {
log("receive exception "+ex.toString());
}
}
mSocket.close();
mSocket = null;
log(methodName+" socket closed.");
}
catch( Exception ex) {
log(methodName+" exception "+ex.toString());
}
}
private static void log( String message) {
Log.d(TAG, message);
}
}
public static boolean unit_test() {
try {
BroadcastSender sender = new BroadcastSender();
sender.start();
BroadcastReceiver receiver = new BroadcastReceiver();
receiver.start();
log( "joining "+sender.getName());
sender.join(TEST_TIME_MS);
log( "joining "+receiver.getName());
receiver.join(TEST_TIME_MS);
boolean success = receiver.totalReceived >= TEST_COUNT;
log( "unit test results="+success);
return success;
}
catch( Exception ex ) {
log( "unit test exception "+ex);
return false;
}
}
/*
* Log output:
*
D 2017-11-17 19:10:56.809 6228 BroadcastSender sent #0 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:57.313 6228 BroadcastSender sent #1 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:57.314 6228 BroadcastReceiver received #1 : "payload pid=6228"
D 2017-11-17 19:10:57.315 6228 BroadcastReceiver received #2 : "payload pid=6228"
D 2017-11-17 19:10:57.816 6228 BroadcastSender sent #2 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:57.817 6228 BroadcastReceiver received #3 : "payload pid=6228"
D 2017-11-17 19:10:57.818 6228 BroadcastReceiver received #4 : "payload pid=6228"
D 2017-11-17 19:10:58.319 6228 BroadcastReceiver received #5 : "payload pid=6228"
D 2017-11-17 19:10:58.319 6228 BroadcastSender sent #3 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:58.321 6228 BroadcastReceiver received #6 : "payload pid=6228"
D 2017-11-17 19:10:58.822 6228 BroadcastSender sent #4 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:58.822 6228 BroadcastReceiver received #7 : "payload pid=6228"
D 2017-11-17 19:10:58.823 6228 BroadcastReceiver received #8 : "payload pid=6228"
D 2017-11-17 19:10:59.325 6228 BroadcastReceiver receive exception java.net.SocketTimeoutException: Receive timed out
D 2017-11-17 19:10:59.326 6228 BroadcastReceiver received #9 : "payload pid=6228"
D 2017-11-17 19:10:59.327 6228 BroadcastReceiver received #10 : "payload pid=6228"
D 2017-11-17 19:10:59.325 6228 BroadcastSender sent #5 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:59.328 6228 BroadcastReceiver run socket closed.
D 2017-11-17 19:10:59.831 6228 BroadcastSender sent #6 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:00.333 6228 BroadcastSender sent #7 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:00.834 6228 BroadcastSender sent #8 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:01.336 6228 BroadcastSender sent #9 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:01.839 6228 BroadcastSender sent #10 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:02.341 6228 BroadcastSender sent #11 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:02.844 6228 BroadcastSender sent #12 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:03.346 6228 BroadcastSender sent #13 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:01.810 6228 BroadcastSender unit test results=true
*/
}
UDP 通信不能依赖于任何类型的进程。问题出在您的代码上。
我的分布式应用程序,有多个主机,需要在同一局域网内找到本地服务器。
我使用 NsdManager 检测本地服务器的成功率有限,这是要走的路,但通常,服务器仍未检测到。我正在寻找更好的解决方案,更可靠的东西。
然后我尝试多播 UDP 数据包,我在同一进程中检测到没有问题。 (我的单元测试同时使用客户端和服务器实例)。那很好用。同样 类 将不允许我的客户端在同一设备上检测到我的服务器 运行(客户端和服务器是单独的进程应用程序)。
然后我改变了使用广播地址的方法,同样,在同一个进程中,单元测试很好,但在两个不同的进程中,找不到服务器。
正如我所说,代码在同一个进程中工作正常,但当 运行 在两个不同的进程中,或在同一 LAN 中的两个不同的客户端/服务器主机上时,它们不起作用。
注意:我的局域网是高端WiFi rooter,没有特殊设置。
有人知道问题出在哪里吗?
请注意,我的清单包括客户端和服务器应用程序的这些设置:
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_NETWORK_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_MULTICAST_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
请注意,为了让我的客户端检测同一主机上的服务器 运行,我已经在使用 TCP 套接字,但我需要检测局域网中的任何服务器。
@Onik 说的很对,
制作一个最小的、完整的、可验证的例子让我找到了问题。
以下是可能感兴趣的人的代码:
package com.example;
import java.net.DatagramPacket;
import java.net.DatagramSocket;
import java.net.InetAddress;
import java.net.InterfaceAddress;
import java.net.NetworkInterface;
import java.util.Enumeration;
import java.util.List;
import android.util.Log;
/**
* Minimal, Complete, and Verifiable example for broadcasting UDP sockets, send thread.
*
* Loosely based on:
*
* https://jackiexie.com/2015/07/15/network-discovery-using-udp-broadcast-java/
*/
public class BroadcastSender extends Thread {
public static final String TAG = "BroadcastSender";
public static final int TEST_PORT_NUMBER = 1234;
public static final int TEST_COUNT = 10;
public static final int DELAY_BETWEEN_PACKETS_MS = 500;
public static final int TEST_TIME_MS = TEST_COUNT * DELAY_BETWEEN_PACKETS_MS;
private DatagramSocket mSocket;
public BroadcastSender() {
super( TAG );
}
public static InetAddress getBroadcastAddress() {
try {
Enumeration<NetworkInterface> interfaces = NetworkInterface.getNetworkInterfaces();
while( interfaces.hasMoreElements() ) {
NetworkInterface current = interfaces.nextElement();
try {
if( !current.isUp() )
continue;
if( current.isLoopback() )
continue;
List<InterfaceAddress> addresses = current.getInterfaceAddresses();
if( addresses == null ) {
log("getBroadcastAddress can not get addresses from "+current);
continue;
}
for( InterfaceAddress oneInterfaceAddress : addresses ) {
InetAddress bcast = oneInterfaceAddress.getBroadcast();
if( bcast != null )
return bcast;
}
}
catch( Exception ex) {
log( "getBroadcastAddress exception "+ex+" stack "+Log.getStackTraceString(ex) );
}
}
}
catch( Exception ex) {
log( "getBroadcastAddress exception "+ex+" stack "+Log.getStackTraceString(ex) );
}
return null;
}
@Override
public void run() {
try {
mSocket = new DatagramSocket();
mSocket.setBroadcast(true);
InetAddress interfaceBroadcast = getBroadcastAddress();
InetAddress broadcastAddr = InetAddress.getByName("255.255.255.255");
String payloadStr = "payload pid="+android.os.Process.myPid();
byte payloadBytes[] = payloadStr.getBytes();
for( int packet = 0; packet < TEST_COUNT; packet++) {
// First send to 255.255.255.255
DatagramPacket dp = new DatagramPacket(payloadBytes, payloadBytes.length, broadcastAddr, TEST_PORT_NUMBER);
mSocket.send(dp);
// Then send to whatever address we have for interfaceBroadcast
if( interfaceBroadcast != null ) {
dp = new DatagramPacket(payloadBytes, payloadBytes.length, interfaceBroadcast, TEST_PORT_NUMBER);
mSocket.send(dp);
}
log("sent #"+packet+" with "+payloadBytes.length+" bytes \""+payloadStr+"\"");
synchronized (this) {
try {
wait(DELAY_BETWEEN_PACKETS_MS);
}
catch( InterruptedException ex) {
log("run exeption "+ex.toString());
}
}
}
mSocket.close();
mSocket = null;
}
catch( Exception ex) {
log("run exception "+ex.toString()+" stack "+Log.getStackTraceString(ex));
}
}
private static void log( String message) {
Log.d(TAG, message);
}
/**
* Minimal, Complete, and Verifiable example for broadcasting UDP sockets, receive thread.
* Loosely based on:
*
* https://jackiexie.com/2015/07/15/network-discovery-using-udp-broadcast-java/
*/
public static class BroadcastReceiver extends Thread {
public static final String TAG = "BroadcastReceiver";
public int totalReceived = 0;
public BroadcastReceiver() {
super( TAG );
}
@Override
public void run() {
final String methodName = "run";
try {
byte messageBytes[] = new byte[1500];
DatagramPacket dp = new DatagramPacket(messageBytes, messageBytes.length);
DatagramSocket mSocket = new DatagramSocket(TEST_PORT_NUMBER);
mSocket.setSoTimeout(DELAY_BETWEEN_PACKETS_MS);
mSocket.setBroadcast(true);
while( totalReceived < TEST_COUNT ) {
try {
// Get one UDP packet
mSocket.receive(dp);
if( dp.getLength() > 0 ) {
totalReceived++;
String payloadStr = new String(dp.getData());
log( "received #"+totalReceived+" : \""+payloadStr+"\"");
}
}
catch( Exception ex) {
log("receive exception "+ex.toString());
}
}
mSocket.close();
mSocket = null;
log(methodName+" socket closed.");
}
catch( Exception ex) {
log(methodName+" exception "+ex.toString());
}
}
private static void log( String message) {
Log.d(TAG, message);
}
}
public static boolean unit_test() {
try {
BroadcastSender sender = new BroadcastSender();
sender.start();
BroadcastReceiver receiver = new BroadcastReceiver();
receiver.start();
log( "joining "+sender.getName());
sender.join(TEST_TIME_MS);
log( "joining "+receiver.getName());
receiver.join(TEST_TIME_MS);
boolean success = receiver.totalReceived >= TEST_COUNT;
log( "unit test results="+success);
return success;
}
catch( Exception ex ) {
log( "unit test exception "+ex);
return false;
}
}
/*
* Log output:
*
D 2017-11-17 19:10:56.809 6228 BroadcastSender sent #0 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:57.313 6228 BroadcastSender sent #1 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:57.314 6228 BroadcastReceiver received #1 : "payload pid=6228"
D 2017-11-17 19:10:57.315 6228 BroadcastReceiver received #2 : "payload pid=6228"
D 2017-11-17 19:10:57.816 6228 BroadcastSender sent #2 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:57.817 6228 BroadcastReceiver received #3 : "payload pid=6228"
D 2017-11-17 19:10:57.818 6228 BroadcastReceiver received #4 : "payload pid=6228"
D 2017-11-17 19:10:58.319 6228 BroadcastReceiver received #5 : "payload pid=6228"
D 2017-11-17 19:10:58.319 6228 BroadcastSender sent #3 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:58.321 6228 BroadcastReceiver received #6 : "payload pid=6228"
D 2017-11-17 19:10:58.822 6228 BroadcastSender sent #4 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:58.822 6228 BroadcastReceiver received #7 : "payload pid=6228"
D 2017-11-17 19:10:58.823 6228 BroadcastReceiver received #8 : "payload pid=6228"
D 2017-11-17 19:10:59.325 6228 BroadcastReceiver receive exception java.net.SocketTimeoutException: Receive timed out
D 2017-11-17 19:10:59.326 6228 BroadcastReceiver received #9 : "payload pid=6228"
D 2017-11-17 19:10:59.327 6228 BroadcastReceiver received #10 : "payload pid=6228"
D 2017-11-17 19:10:59.325 6228 BroadcastSender sent #5 with 16 bytes "payload pid=6228"
D 2017-11-17 19:10:59.328 6228 BroadcastReceiver run socket closed.
D 2017-11-17 19:10:59.831 6228 BroadcastSender sent #6 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:00.333 6228 BroadcastSender sent #7 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:00.834 6228 BroadcastSender sent #8 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:01.336 6228 BroadcastSender sent #9 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:01.839 6228 BroadcastSender sent #10 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:02.341 6228 BroadcastSender sent #11 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:02.844 6228 BroadcastSender sent #12 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:03.346 6228 BroadcastSender sent #13 with 16 bytes "payload pid=6228"
D 2017-11-17 19:11:01.810 6228 BroadcastSender unit test results=true
*/
}
UDP 通信不能依赖于任何类型的进程。问题出在您的代码上。