Определить, есть ли у Android-устройства подключение к Интернету - PullRequest
130 голосов
/ 27 июня 2011

Мне нужно сказать, есть ли у моего устройства подключение к Интернету или нет.Я нашел много ответов, таких как:

private boolean isNetworkAvailable() {
    ConnectivityManager connectivityManager 
         = (ConnectivityManager) getSystemService(Context.CONNECTIVITY_SERVICE);
    NetworkInfo activeNetworkInfo = connectivityManager.getActiveNetworkInfo();
    return activeNetworkInfo != null;
}

(взято из Определить, есть ли доступное подключение к Интернету на Android .)

Но этоэто неправильно, например, если я подключен к беспроводной сети, которая не имеет доступа к Интернету , этот метод вернет true… Есть ли способ определить, есть ли у устройства подключение к Интернетуа нет, если он только к чему-то подключен?

Ответы [ 9 ]

180 голосов
/ 27 июня 2011

Вы правы. Код, который вы предоставили, только проверяет наличие сетевого подключения. Лучший способ проверить, есть ли активное подключение к Интернету, это попытаться подключиться на известный сервер через http.

public static boolean hasActiveInternetConnection(Context context) {
    if (isNetworkAvailable(context)) {
        try {
            HttpURLConnection urlc = (HttpURLConnection) (new URL("http://www.google.com").openConnection());
            urlc.setRequestProperty("User-Agent", "Test");
            urlc.setRequestProperty("Connection", "close");
            urlc.setConnectTimeout(1500); 
            urlc.connect();
            return (urlc.getResponseCode() == 200);
        } catch (IOException e) {
            Log.e(LOG_TAG, "Error checking internet connection", e);
        }
    } else {
        Log.d(LOG_TAG, "No network available!");
    }
    return false;
}

Конечно, вы можете заменить http://www.google.com URL-адрес любым другим сервером, к которому вы хотите подключиться, или сервер, который, как вы знаете, имеет хорошее время работы.

Как Тони Чо также указал в этом комментарии ниже , убедитесь, что вы не запускаете этот код в главном потоке, в противном случае вы получите исключение NetworkOnMainThread (в Android 3.0 или более поздней версии). Вместо этого используйте AsyncTask или Runnable.

Если вы хотите использовать google.com, обратите внимание на модификацию Jeshurun. В своем ответе он изменил мой код и сделал его немного более эффективным. Если вы подключитесь к

HttpURLConnection urlc = (HttpURLConnection) 
            (new URL("http://clients3.google.com/generate_204")
            .openConnection());

и затем проверьте код ответа для 204

return (urlc.getResponseCode() == 204 && urlc.getContentLength() == 0);

тогда вам не нужно сначала извлекать всю главную страницу Google.

86 голосов
/ 13 сентября 2014

Я немного изменил ответ THelper, чтобы использовать известный взлом, который Android уже использует, чтобы проверить, есть ли у подключенной сети Wi-Fi доступ в Интернет.Это намного эффективнее, чем захват всей домашней страницы Google.См. здесь и здесь для получения дополнительной информации.

public static boolean hasInternetAccess(Context context) {
    if (isNetworkAvailable(context)) {
        try {
            HttpURLConnection urlc = (HttpURLConnection) 
                (new URL("http://clients3.google.com/generate_204")
                .openConnection());
            urlc.setRequestProperty("User-Agent", "Android");
            urlc.setRequestProperty("Connection", "close");
            urlc.setConnectTimeout(1500); 
            urlc.connect();
            return (urlc.getResponseCode() == 204 &&
                        urlc.getContentLength() == 0);
        } catch (IOException e) {
            Log.e(TAG, "Error checking internet connection", e);
        }
    } else {
        Log.d(TAG, "No network available!");
    }
    return false;
}
16 голосов
/ 09 июня 2015
public boolean isInternetWorking() {
    boolean success = false;
    try {
        URL url = new URL("https://google.com");
        HttpURLConnection connection = (HttpURLConnection) url.openConnection();
        connection.setConnectTimeout(10000);
        connection.connect();
        success = connection.getResponseCode() == 200;
    } catch (IOException e) {
        e.printStackTrace();
    }
    return success;
}

вернуть true, если интернет действительно доступен

Убедитесь, что у вас есть эти два разрешения

<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
10 голосов
/ 26 января 2016

Если вы нацелены на Lollipop или выше, можно использовать новый класс NetworkCapabilities, например:

public static boolean hasInternetConnection(final Context context) {
    final ConnectivityManager connectivityManager = (ConnectivityManager)context.
            getSystemService(Context.CONNECTIVITY_SERVICE);

    final Network network = connectivityManager.getActiveNetwork();
    final NetworkCapabilities capabilities = connectivityManager
            .getNetworkCapabilities(network);

    return capabilities != null
            && capabilities.hasCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED);
}
5 голосов
/ 26 февраля 2016

Вам не обязательно устанавливать полное HTTP-соединение. Вы можете попробовать открыть TCP-соединение с известным хостом, и, если это удастся, у вас будет интернет-соединение.

public boolean hostAvailable(String host, int port) {
  try (Socket socket = new Socket()) {
    socket.connect(new InetSocketAddress(host, port), 2000);
    return true;
  } catch (IOException e) {
    // Either we have a timeout or unreachable host or failed DNS lookup
    System.out.println(e);
    return false;
  }
}

Тогда просто проверьте с помощью:

boolean online = hostAvailable("www.google.com", 80);
3 голосов
/ 03 февраля 2017

Основываясь на принятых ответах, я построил этот класс со слушателем, чтобы вы могли использовать его в основном потоке:

Первый : класс InterntCheck, который проверяет подключение к Интернету в фоновом режиме, а затем вызывает метод слушателя с результатом.

public class InternetCheck extends AsyncTask<Void, Void, Void> {


    private Activity activity;
    private InternetCheckListener listener;

    public InternetCheck(Activity x){

        activity= x;

    }

    @Override
    protected Void doInBackground(Void... params) {


        boolean b = hasInternetAccess();
        listener.onComplete(b);

        return null;
    }


    public void isInternetConnectionAvailable(InternetCheckListener x){
        listener=x;
        execute();
    }

    private boolean isNetworkAvailable() {
        ConnectivityManager connectivityManager = (ConnectivityManager) activity.getSystemService(CONNECTIVITY_SERVICE);
        NetworkInfo activeNetworkInfo = connectivityManager.getActiveNetworkInfo();
        return activeNetworkInfo != null;
    }
    private boolean hasInternetAccess() {
        if (isNetworkAvailable()) {
            try {
                HttpURLConnection urlc = (HttpURLConnection) (new URL("http://clients3.google.com/generate_204").openConnection());
                urlc.setRequestProperty("User-Agent", "Android");
                urlc.setRequestProperty("Connection", "close");
                urlc.setConnectTimeout(1500);
                urlc.connect();
                return (urlc.getResponseCode() == 204 &&
                        urlc.getContentLength() == 0);
            } catch (IOException e) {
                e.printStackTrace();
            }
        } else {
            Log.d("TAG", "No network available!");
        }
        return false;
    }

    public interface InternetCheckListener{
        void onComplete(boolean connected);
    }

}

Second : создать экземпляр класса в главном потоке и дождаться ответа (если вы работали с API Firebase для Android, прежде чем это должно быть вам знакомо!).

new InternetCheck(activity).isInternetConnectionAvailable(new InternetCheck.InternetCheckListener() {

        @Override
        public void onComplete(boolean connected) {
           //proceed!
        }
    });

Теперь внутри метода onComplete вы узнаете, подключено ли устройство к Интернету или нет.

3 голосов
/ 10 июля 2015

попробуйте это

public class ConnectionDetector {
    private Context _context;

    public ConnectionDetector(Context context) {
        this._context = context;
    }

    public boolean isConnectingToInternet() {
        if (networkConnectivity()) {
            try {
                HttpURLConnection urlc = (HttpURLConnection) (new URL(
                        "http://www.google.com").openConnection());
                urlc.setRequestProperty("User-Agent", "Test");
                urlc.setRequestProperty("Connection", "close");
                urlc.setConnectTimeout(3000);
                urlc.setReadTimeout(4000);
                urlc.connect();
                // networkcode2 = urlc.getResponseCode();
                return (urlc.getResponseCode() == 200);
            } catch (IOException e) {
                return (false);
            }
        } else
            return false;

    }

    private boolean networkConnectivity() {
        ConnectivityManager cm = (ConnectivityManager) _context
                .getSystemService(Context.CONNECTIVITY_SERVICE);
        NetworkInfo networkInfo = cm.getActiveNetworkInfo();
        if (networkInfo != null && networkInfo.isConnected()) {
            return true;
        }
        return false;
    }
}

вам нужно добавить следующее разрешение в ваш файл манифеста:

<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.INTERNET" />

Затем позвоните какчто:

if((new ConnectionDetector(MyService.this)).isConnectingToInternet()){
    Log.d("internet status","Internet Access");
}else{
    Log.d("internet status","no Internet Access");
}
0 голосов
/ 23 сентября 2018

Самый последний способ сделать это из документации - это использовать ConnectivityManager для запроса активной сети и определения наличия подключения к Интернету.

public boolean hasInternetConnectivity() {
    ConnectivityManager cm =
        (ConnectivityManager)context.getSystemService(Context.CONNECTIVITY_SERVICE);

    NetworkInfo activeNetwork = cm.getActiveNetworkInfo();
    return (activeNetwork != null &&
                      activeNetwork.isConnectedOrConnecting());
}

Добавьте эти дваразрешения для вашего файла AndroidManifest.xml:

<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE"/>
0 голосов
/ 14 февраля 2017
private static NetworkUtil mInstance;
private volatile boolean mIsOnline;

private NetworkUtil() {
    ScheduledExecutorService exec = Executors.newSingleThreadScheduledExecutor();
    exec.scheduleAtFixedRate(new Runnable() {
        @Override
        public void run() {
            boolean reachable = false;
            try {
                Process process = java.lang.Runtime.getRuntime().exec("ping -c 1 www.google.com");
                int returnVal = process.waitFor();
                reachable = (returnVal==0);
            } catch (Exception e) {
                e.printStackTrace();
            }
            mIsOnline = reachable;
        }
    }, 0, 5, TimeUnit.SECONDS);
}

public static NetworkUtil getInstance() {
    if (mInstance == null) {
        synchronized (NetworkUtil.class) {
            if (mInstance == null) {
                mInstance = new NetworkUtil();
            }
        }
    }
    return mInstance;
}

public boolean isOnline() {
    return mIsOnline;
}

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

...