Как я могу получить уведомление, когда устройство теряет подключение к сети?

Я знаю, что вы можете написать код, с помощью которого вы можете определить, подключено ли устройство к сети.

Но в моем приложении я хочу получить уведомление, если устройство меняет свое состояние с "сеть" на "нет сети". Это может произойти, например, когда пользователь входит в туннель и теряет сигнал и т. Д. Или когда он подключен к сети 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
    }

Сделайте это в потоке.

Другие вопросы по тегам