Как я могу получить уведомление, когда устройство теряет подключение к сети?
Я знаю, что вы можете написать код, с помощью которого вы можете определить, подключено ли устройство к сети.
Но в моем приложении я хочу получить уведомление, если устройство меняет свое состояние с "сеть" на "нет сети". Это может произойти, например, когда пользователь входит в туннель и теряет сигнал и т. Д. Или когда он подключен к сети Wi-Fi, пользователь выходит за пределы зоны доступа и больше не имеет доступа к Интернету.
Предоставляет ли Android API что-то, где вы можете зарегистрировать прослушиватель, чтобы получать уведомления каждый раз, когда происходит изменение состояния сети?
Я нашел этот код и попытался его использовать, но он ничего не делает. Я не получаю никаких уведомлений при изменении состояния сети.
public class ConnectivityManager extends PhoneStateListener{
Activity activity;
public ConnectivityManager(Activity a){
TelephonyManager telephonyManager = (TelephonyManager)a.getSystemService(Context.TELEPHONY_SERVICE);
telephonyManager.listen(this, PhoneStateListener.LISTEN_DATA_CONNECTION_STATE);
activity = a;
}
@Override
public void onDataConnectionStateChanged(int state) {
super.onDataConnectionStateChanged(state);
switch (state) {
case TelephonyManager.DATA_DISCONNECTED:
new AlertDialog.Builder(activity).
setCancelable(false).
setTitle("Connection Manager").
setMessage("There is no network connection. Please connect to internet and start again.").
setNeutralButton("Ok", new DialogInterface.OnClickListener() {
public void onClick(DialogInterface dialog, int whichButton) {
System.exit(0);
}
}).create();
break;
case TelephonyManager.DATA_CONNECTED:
break;
}
}
}
Также я добавил соответствующие разрешения в AndroidManifest.xml:
<uses-permission android:name="android.permission.INTERNET"></uses-permission>
<uses-permission android:name="android.permission.READ_PHONE_STATE"></uses-permission>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"></uses-permission>
5 ответов
myTelephonyManager=(TelephonyManager)getSystemService(Context.TELEPHONY_SERVICE);
callStateListener = new PhoneStateListener(){
public void onDataConnectionStateChanged(int state){
switch(state){
case TelephonyManager.DATA_DISCONNECTED:
Log.i("State: ", "Offline");
// String stateString = "Offline";
// Toast.makeText(getApplicationContext(),
// stateString, Toast.LENGTH_LONG).show();
break;
case TelephonyManager.DATA_SUSPENDED:
Log.i("State: ", "IDLE");
// stateString = "Idle";
// Toast.makeText(getApplicationContext(),
// stateString, Toast.LENGTH_LONG).show();
break;
}
}
};
myTelephonyManager.listen(callStateListener,
PhoneStateListener.LISTEN_DATA_CONNECTION_STATE);
Вы можете рассмотреть возможность использования BroadcastReceiver
за ConnectivityManager.CONNECTIVITY_ACTION
вместо. Из документов:
Произошло изменение подключения к сети. Соединение было установлено или потеряно.NETworkInfo для уязвимой сети отправляется как дополнительная; следует проконсультироваться, чтобы увидеть, какое событие подключения произошло.
Этот приемник работает как для WiFi, так и для сотовой связи, в отличие от PhoneStateListener.LISTEN_DATA_CONNECTION_STATE
, который будет только уведомлять вас об изменениях в сотовых сетях.
Принятый ответ не работал для меня (мой планшет был подключен к Wi-Fi - нет 3G). Следующий код работал для меня:
public class ConnectivityChangeReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
System.out.println("*** Action: " + intent.getAction());
if(intent.getAction().equalsIgnoreCase("android.net.conn.CONNECTIVITY_CHANGE")) {
Toast.makeText(context, "Connection changed", Toast.LENGTH_SHORT).show();
}
}
}
и изменение в AndroidManifest.xml
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<receiver android:name=".custom.ConnectivityChangeReceiver">
<intent-filter>
<action android:name="android.net.conn.CONNECTIVITY_CHANGE" />
</intent-filter>
</receiver>
Я реализовал эту функцию путем опроса ConnectivityManager.
_connectivityManager = (ConnectivityManager)getSystemService(CONNECTIVITY_SERVICE);
Как только у нас есть менеджер по подключению, просто опрашивайте его время от времени, чтобы увидеть, если вы все еще подключены
private final ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
final Runnable networkStatusChecker = new Runnable() {
public void run() {
NetworkInfo activeNetwork = _connectivityManager.getActiveNetworkInfo();
boolean isConnected = activeNetwork != null && activeNetwork.isConnectedOrConnecting();
if(!isConnected) {
System.out.println("disconnected from network");
networkCheckerHandle.cancel(true);
}
}
};
final ScheduledFuture networkCheckerHandle = scheduler.scheduleAtFixedRate(networkStatusChecker, 10, 10, SECONDS);
Вы можете легко использовать Мерлина и его бороду. Здесь вы также можете получить тип подключения: мобильный или Wi-Fi.
implementation 'com.novoda:merlin:1.2.0'
private MerlinsBeard merlinsBeard;
private NetworkChecker(Context context){
Merlin merlin = new Merlin.Builder()
.withConnectableCallbacks()
.withDisconnectableCallbacks()
.build(context);
merlinsBeard = new MerlinsBeard.Builder().build(context);
merlin.registerConnectable(() -> {
Timber.d("Connection enabled");
});
merlin.registerDisconnectable(() -> {
Timber.d("Connection disabled");
});
merlin.bind();
}
public boolean getConnected(){
return merlinsBeard.isConnected();
}
Если вы пытаетесь сделать это в Котлине. Использовать этот
val isInternetWorking: Boolean by lazy {
var success = false
try {
val url = URL("https://google.com")
val connection = url.openConnection() as HttpURLConnection
connection.connectTimeout = 10000
connection.connect()
success = connection.responseCode === 200
} catch (e: IOException) {
e.printStackTrace()
}
return@lazy success
}
Сделайте это в потоке.