Próbuję opracować aplikację, która wysyła niektóre wiadomości rozgłoszeniowe i otrzymuje odpowiedzi z innych urządzeń z Androidem. Mam problem z odbieraniem wiadomości UDP z innych urządzeń. Powinienem wspomnieć, że ten kod działał na Gingerbread, ale na JellyBean to już nie działa i nie wiem, jaki może być problem.Wysyłaj transmisję UDP, ale nie odbieraj jej na innych urządzeniach z Androidem.
Tutaj jest gdzie wysłać komunikat rozgłoszeniowy (znam inne urządzenia nasłuchuje na porcie 5000):
private void sendUDPMessage(String msg) {
try {
DatagramSocket clientSocket = new DatagramSocket();
clientSocket.setBroadcast(true);
InetAddress address = InetAddress.getByName(Utils.getBroadcastAddress());
byte[] sendData;
sendData = msg.getBytes();
DatagramPacket sendPacket = new DatagramPacket(sendData,
sendData.length, address, 5000);
clientSocket.send(sendPacket);
clientSocket.close();
} catch (Exception e) {
e.printStackTrace();
}
}
A oto gdzie otrzymam go:
private void start_UDP()
{
try {
serverSocketUDP = new DatagramSocket(5000);
}
catch (Exception e) {
Log.i(LOGTAG, "Exception opening DatagramSocket UDP");
}
final byte[] receiveData = new byte[1024];
while(runningUDP) {
Log.d(LOGTAG, "Waiting for Broadcast request in ServerUDP.");
final DatagramPacket receivePacket = new DatagramPacket(receiveData, receiveData.length);
serverSocketUDP.receive(receivePacket);
byte[] sendData = new byte[1024];
InetAddress address = receivePacket.getAddress();
int port = receivePacket.getPort();
if(!receivePacket.getAddress().getHostAddress().equals(Utils.getLocalIpAddress()))
{
String req = new String(receivePacket.getData(), 0, receivePacket.getLength());
Log.d(LOGTAG, "Received UDP message : "+req+" from: "+receivePacket.getAddress().getHostAddress());
}
}// while ends
}//method ends
należy wspomnieć, że te 2 funkcje są oddzielne w 2 różnych wątkach, więc mogę wysyłać i odbierać jednocześnie.
ja również nabyć następujące zamki:
powerManager =(PowerManager)context.getSystemService(Context.POWER_SERVICE);
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK ,LOGTAG); // PARTIAL_WAKE_LOCK Only keeps CPU on
wifiManager = (WifiManager)context.getSystemService(Context.WIFI_SERVICE);
wifiLock = wifiManager.createWifiLock(3, LOGTAG);
multicastLock = wifiManager.createMulticastLock(LOGTAG);
wakeLock.acquire();
multicastLock.acquire();
wifiLock.acquire();
A uprawnienia do pliku manifestu:
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.WRITE_SETTINGS"/>
<uses-permission android:name="android.permission.CHANGE_WIFI_MULTICAST_STATE"/>
I sprawdzili, czy wiadomości są wysyłane przy użyciu Wireshark i tcpdump i ich wysłania. Co więcej, co jeszcze dziwniejsze, odbieram wysyłane wiadomości, które wysyłam (ale odrzucam je, ponieważ nie muszę przetwarzać wiadomości wysłanych ode mnie), ale nie odbieram wiadomości rozgłoszeniowych wysyłanych z innych urządzeń (które powinny mieć ten sam format, tylko adres źródłowy byłby inny, a wiadomość zawarta w żaden sposób nie powinna wpływać na komunikat rozgłoszeniowy).
Proszę dać mi znać, jeśli masz jakieś pomysły, bo zabrakło mi wszystkiego, co mógłbym spróbować. Każda pomoc będzie doceniona. Dzięki!
EDIT: Zrobiłem kilka testów, a nawet jeśli, gdy biegnę na każdym z wlan0 telefony ifconfig i mówi coś
ifconfig wlan0
wlan0: ip 169.254.17.28 mask 255.255.0.0 flags [up broadcast multicast]
co oznacza, że interfejs jest aktywny, a IP jest ustawiony, może odbierać wiadomości rozgłoszeniowe i wiadomości multiemisyjne, ale gdy używam
InetAddress in=InetAddress.getByName("169.254.17.28");
if (in.isReachable(1000))
Log.i(LOGTAG, "host is reachable");
else
Log.i(LOGTAG, "host is not reachable");
To pokazuje, że w logi host nie jest osiągalny.
To gdzie mam włączyć Wi-Fi
private void startWifiAdhoc() {
WifiManager wifiManager = (WifiManager)SharingFileService.context.getSystemService(Context.WIFI_SERVICE);
String command="";
if (condWifiAdhoc == false) {
condWifiAdhoc=true;
wifiInterface = Utils.getWifiInterface();
wifiManager.setWifiEnabled(true);
localIP = Utils.getLinkLocalAddress();
}
else
{
wifiManager.setWifiEnabled(true);
localIP = Utils.getLinkLocalAddress();
}
// Set wifi ad-hoc
command = context.getFilesDir().getPath()
+ "/iwconfig " + wifiInterface + " mode ad-hoc essid "
+ "mcp" + " channel " + "1" + " commit\n";
Log.i(LOGTAG, command);
Utils.rootExec(command);
Log.i(LOGTAG, "Ip address used :" + localIP);
command = context.getFilesDir().getPath()
+ "/ifconfig " + wifiInterface + " " + localIP
+ " netmask 255.255.0.0 up\n";
Log.i(LOGTAG, command);
Utils.rootExec(command);
}
pamiętać, że niektóre routery wyłączają domyślnie DNS rozsyłania grupowego. – yorkw
oh ... Muszę wspomnieć, że jestem w WiFi adhoc – Lara
Możesz chcieć spojrzeć na ten post: http://stackoverflow.com/a/16208617/1028256 Mówi, że niektórzy kierowcy wifi mogą wyłączyć odbiorniki, ale w takim przypadku dzieje się to po wznowieniu z trybu uśpienia. – Mixaz