android 监听网络状态的变化及实战,安卓开发入门教程

慈云数据 2024-05-13 技术支持 81 0
  • Monitor network connections (Wi-Fi, GPRS, UMTS, etc.) (用来处理网络连接 ,包括Wi-Fi, GPRS, UMTS等)

  • Send broadcast intents when network connectivity changes(用 网络状态发生变化的时候发出 广播 )

  • Attempt to “fail over” to another network when connectivity to a network is lost(但断开网络连接的时候,尝试去;连接另外一个网络

  • Provide an API that allows applications to query the coarse-grained or fine-grained state of the available networks(

  • Provide an API that allows applications to request and select networks for their data traffic

    怎样获取ConnectivityManager对象呢?

    ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context

    .CONNECTIVITY_SERVICE);

    获取 NetworkInfo对象

    主要有一下 几种方法

    • getNetworkInfo(int networkType),但是这个方法已经过时,官网的解释如下:This method was deprecated in API level 23. This method does not support multiple connected networks of the same type. Use getAllNetworks() and getNetworkInfo(android.net.Network) instead.

    • getNetworkInfo(Network network)

    • getActiveNetwork()

      Returns a Network object corresponding to the currently active default data network.

      • getActiveNetworkInfo(),Returns details about the currently active default data network.

      • getAllNetworkInfo()这个方法已经过时,Use getAllNetworks() and getNetworkInfo(android.net.Network) instead.

        综上所述,我们如果要知道当前Mobile网络或者WiFi网络是否已经连接上,总共有两种方法。

        第一种方法,只不过在 API23的时候已经 过时了

        State wifiState = null;

        State mobileState = null;

        ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context

        .CONNECTIVITY_SERVICE);

        wifiState = cm.getNetworkInfo(ConnectivityManager.TYPE_WIFI).getState();

        mobileState = cm.getNetworkInfo(ConnectivityManager.TYPE_MOBILE).getState();

        Log.d(TAG1,

        “wifi状态:” + wifiState + “\n mobile状态:” + mobileState);

        if (wifiState != null && mobileState != null

        && State.CONNECTED != wifiState

        && State.CONNECTED == mobileState) {// 手机网络连接成功

        Log.d(TAG1, “手机2g/3g/4g网络连接成功”);

        APP.getInstance().setMobile(true);

        APP.getInstance().setWifi(false);

        APP.getInstance().setConnected(true);

        } else if (wifiState != null && State.CONNECTED == wifiState) {// 无线网络连接成功

        Log.d(TAG1, “无线网络连接成功”);

        APP.getInstance().setMobile(false);

        APP.getInstance().setWifi(true);

        APP.getInstance().setConnected(true);

        } else if (wifiState != null && mobileState != null

        && State.CONNECTED != wifiState

        && State.CONNECTED != mobileState) {// 手机没有任何的网络

        Log.d(TAG1, “手机没有任何的网络”);

        APP.getInstance().setMobile(false);

        APP.getInstance().setWifi(false);

        APP.getInstance().setConnected(false);

        }

        第二种方法

        ConnectivityManager manager = (ConnectivityManager) context

        .getSystemService(Context.CONNECTIVITY_SERVICE);

        Log.i(TAG1, “CONNECTIVITY_ACTION”);

        NetworkInfo activeNetwork = manager.getActiveNetworkInfo();

        if (activeNetwork != null) { // connected to the internet

        if (activeNetwork.isConnected()) {

        if (activeNetwork.getType() == ConnectivityManager.TYPE_WIFI) {

        // connected to wifi

        APP.getInstance().setWifi(true);

        Log.e(TAG, "当前WiFi连接可用 ");

        } else if (activeNetwork.getType() == ConnectivityManager.TYPE_MOBILE) {

        // connected to the mobile provider’s data plan

        APP.getInstance().setMobile(true);

        Log.e(TAG, "当前移动网络连接可用 ");

        }

        } else {

        Log.e(TAG, "当前没有网络连接,请确保你已经打开网络 ");

        }

        } else { // not connected to the internet

        Log.e(TAG, "当前没有网络连接,请确保你已经打开网络 ");

        APP.getInstance().setWifi(false);

        APP.getInstance().setMobile(false);

        APP.getInstance().setConnected(false);

        }


        网络错误情况的处理


        正如前面所提到的,这篇博客吧网络错误主要分为两大类

        • 没有网络情况的错误

        • 在有网络的情况下,我们客户端的错误或者服务器端的错误

          在这里 我们主要处理没有网络情况下的错误,现在 个人了解到 的主要有两种处理方法。

          第一种做法:

          在APP启动的 时候检查当前是否已经连接上网络,弹出一个对话框没有的话跳转到设置界面或者WiFi设置界面或者打开移动网络界面 。

          第二种方法

          其实跟第一种做法差不多,只是在每一次 错误的情况下,都会判断当前有没有 网络 ,没有弹出一个对话框,跳转到设置界面或者WiFi设置界面或者打开移动网络界面 , 下面我们我们一起来看一下 怎样弹出一个对话框,并且跳转到相应的设置界面

          这里我们采取第一种做法,效果图如下

          [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-soHQMd4i-1617723276312)(https://raw.githubusercontent.com/gdutxiaoxu/blog_pic/master/19_05/9fe4afa0gw1f9dp8oqfc2g208u0gjh6x.gif)]

          代码如下

          public static void showWifiDlg(final Context context) {

          AlertDialog.Builder builder = new AlertDialog.Builder(context.getApplicationContext());

          if (mWifiDialog == null) {

          mWifiDialog = builder.setIcon(R.drawable.ic_launcher) //

          .setTitle(“wifi设置”) //

          .setMessage(“当前无网络”).setPositiveButton(“设置”, new DialogInterface

          .OnClickListener() {

          @Override

          public void onClick(DialogInterface dialog, int which) {

          // 跳转到系统的网络设置界面

          Intent intent = null;

          // 先判断当前系统版本

          if (android.os.Build.VERSION.SDK_INT > 10) { // 3.0以上

          intent = new Intent(android.provider.Settings

          .ACTION_WIFI_SETTINGS);

          } else {

          intent = new Intent();

          intent.setClassName(“com.android.settings”,

          Settings.ACTION_WIFI_SETTINGS);

          }

          if ((context instanceof Application)) {

          intent.addFlags(FLAG_ACTIVITY_NEW_TASK);

          }

          context.startActivity(intent);

          }

          }).setNegativeButton(“知道了”, null).create();

          // 设置为系统的Dialog,这样使用Application的时候不会 报错

          mWifiDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);

          }

          mWifiDialog.show();

          }

          这里对几个 重要的 Action说一下

          • ACTION_DATA_ROAMING_SETTINGS : 跳转到移动网络设置界面

          • ACTION_WIFI_SETTINGS

            Activity Action: Show settings to allow configuration of Wi-Fi.

            • ACTION_WIRELESS_SETTINGS

              Activity Action: Show settings to allow configuration of wireless controls such as Wi-Fi, Bluetooth and Mobile networks.

              关于更多Activity Action,请参考官网地址

              需要注意的是

              若我们 使用的Context不是Activity 的Context 而是Application的 Context,我们 需要做以下处理 ,否则会报错

              // 设置为系统级别的Dialog

              mWifiDialog.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);

              if ((context instanceof Application)) { intent.addFlags(FLAG_ACTIVITY_NEW_TASK);

              }

              context.startActivity(intent);

              在AndroidMainFest中添加以下权限 。


              智能无图


              智能无图,这种浏览模式我们平时 很常见,比如在UC浏览器中,网易新闻中都有看到这种模式,这种模式的实质就是监听网络状态,再根据是否是WiFi去确定是否加载网络图片。

              效果图如下

              我们可以看到在开启智能无图的情况下,若不是连接WiFi ,我们是不会去加载网络图片的。

              实现的核心代码如下

              1)当智能无图模式变化的时候,我们会把标志存进sharepreferences

              mSwitchWifiPic.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {

              @Override

              public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {

              SPUtils.put(SPConstants.isIntelligentNoPic,isChecked);

              }

              });

              同时我们为了进来的时候界面与SharePreferences中的 isIntelligentNoPic的值保持一致,我们需要调用一下代码

              boolean isIntelligentNoPic = SPUtils.getBoolean(SPConstants.isIntelligentNoPic);

              mSwitchWifiPic.setChecked(isIntelligentNoPic);

              2)在NewsListAdapter中

              // 是否开启智能无图模式,true表示开启智能无图模式

              boolean isIntelligentNoPic = SPUtils.getBoolean(Constants.SPConstants.isIntelligentNoPic);

              WriteLogUtil.i(“isIntelligentNoPic=” + isIntelligentNoPic);

              if (isIntelligentNoPic) {

              if (APP.getInstance().isWifi()) {

              GlideUtils.display(mContext, iv, picUrl);

              } else {

              iv.setImageDrawable(new ColorDrawable(Color.GRAY));

              }

              } else {

              GlideUtils.display(mContext, iv, picUrl);

              }

              至于APP.getInstance().isWifi()表示当前是否连接WiFi,是通过 监听广播实现的 ,前面已经说明了,这里就不再阐述了。


              拓展工具类


              以下工具类参考Android获取网络状态

              public class NetStateUtils {

              /**

              • 判断当前网络是否是移动网络

              • @param context

              • @return boolean

                */

                public static boolean is3G(Context context) {

                ConnectivityManager connectivityManager = (ConnectivityManager) context

                .getSystemService(Context.CONNECTIVITY_SERVICE);

                NetworkInfo activeNetInfo = connectivityManager.getActiveNetworkInfo();

                if (activeNetInfo != null

                && activeNetInfo.getType() == ConnectivityManager.TYPE_MOBILE) {

                return true;

                }

                return false;

                }

                /**

                • 判断当前网络是否是wifi网络

                • @param context

                • @return boolean

                  */

                  public static boolean isWifi(Context context) {

                  ConnectivityManager connectivityManager = (ConnectivityManager) context

                  .getSystemService(Context.CONNECTIVITY_SERVICE);

                  NetworkInfo activeNetInfo = connectivityManager.getActiveNetworkInfo();

                  if (activeNetInfo != null

                  && activeNetInfo.getType() == ConnectivityManager.TYPE_WIFI) {

                  return true;

                  }

                  return false;

                  }

                  /**

                  • 判断当前网络是否是2G网络

                  • @param context

                  • @return boolean

                    */

                    public static boolean is2G(Context context) {

                    ConnectivityManager connectivityManager = (ConnectivityManager) context

                    .getSystemService(Context.CONNECTIVITY_SERVICE);

                    NetworkInfo activeNetInfo = connectivityManager.getActiveNetworkInfo();

                    if (activeNetInfo != null

                    && (activeNetInfo.getSubtype() == TelephonyManager.NETWORK_TYPE_EDGE

                    || activeNetInfo.getSubtype() == TelephonyManager.NETWORK_TYPE_GPRS || activeNetInfo

                    .getSubtype() == TelephonyManager.NETWORK_TYPE_CDMA)) {

                    return true;

                    }

                    return false;

                    }

                    /**

                    • wifi是否打开

                      */

                      public static boolean isWifiEnabled(Context context) {

                      ConnectivityManager mgrConn = (ConnectivityManager) context

                      .getSystemService(Context.CONNECTIVITY_SERVICE);

                      TelephonyManager mgrTel = (TelephonyManager) context

                      .getSystemService(Context.TELEPHONY_SERVICE);

                      return ((mgrConn.getActiveNetworkInfo() != null && mgrConn

                      .getActiveNetworkInfo().getState() == NetworkInfo.State.CONNECTED) || mgrTel

                      .getNetworkType() == TelephonyManager.NETWORK_TYPE_UMTS);

                      }

                      /**

                      • 判断是否有网络连接

                      • @param context

                      • @return

                        */

                        public static boolean isNetworkConnected(Context context) {

                        if (context != null) {

                        // 获取手机所有连接管理对象(包括对wi-fi,net等连接的管理)

                        ConnectivityManager manager = (ConnectivityManager) context.getSystemService(Context

                        .CONNECTIVITY_SERVICE);

                        // 获取NetworkInfo对象

                        NetworkInfo networkInfo = manager.getActiveNetworkInfo();

                        //判断NetworkInfo对象是否为空

                        if (networkInfo != null)

                        return networkInfo.isAvailable();

                        }

                        return false;

                        }

                        /**

                        • 判断MOBILE网络是否可用

                        • @param context

                        • @param context

                        • @return

                          */

                          public static boolean isMobileConnected(Context context) {

                          if (context != null) {

                          //获取手机所有连接管理对象(包括对wi-fi,net等连接的管理)

                          ConnectivityManager manager = (ConnectivityManager) context.getSystemService(Context

                          .CONNECTIVITY_SERVICE);

                          //获取NetworkInfo对象

                          NetworkInfo networkInfo = manager.getActiveNetworkInfo();

                          //判断NetworkInfo对象是否为空 并且类型是否为MOBILE

                          if (networkInfo != null && networkInfo.getType() == ConnectivityManager.TYPE_MOBILE)

                          return networkInfo.isAvailable();

                          }

                          return false;

                          }

                          /**

                          • 获取当前网络连接的类型信息

                          • 原生

                          • @param context

                          • @return

                            */

                            public static int getConnectedType(Context context) {

                            if (context != null) {

                            //获取手机所有连接管理对象

                            自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

                            深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

                            因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

                            img

                            img

                            img

                            img

                            img

                            img

                            img

                            既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

                            由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

                            如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)

                            img

                            最后

                            感觉现在好多人都在说什么安卓快凉了,工作越来越难找了。又是说什么程序员中年危机啥的,为啥我这年近30的老农根本没有这种感觉,反倒觉得那些贩卖焦虑的都是瞎j8扯谈。当然,职业危机意识确实是要有的,但根本没到那种草木皆兵的地步好吗?

                            Android凉了都是弱者的借口和说辞。虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。

                            所以,最后这里放上我耗时两个月,将自己8年Android开发的知识笔记整理成的Android开发者必知必会系统学习资料笔记,上述知识点在笔记中都有详细的解读,里面还包含了腾讯、字节跳动、阿里、百度2019-2021面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。

                            以上全套学习笔记面试宝典,吃透一半保你可以吊打面试官,只有自己真正强大了,有核心竞争力,你才有拒绝offer的权力,所以,奋斗吧!骚年们!千里之行,始于足下。种下一颗树最好的时间是十年前,其次,就是现在。

                            最后,赠与大家一句诗,共勉!

                            不驰于空想,不骛于虚声。不忘初心,方得始终。

                            正体系化!**

                            由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

                            如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)

                            [外链图片转存中…(img-i7au9Am6-1711730578956)]

                            最后

                            感觉现在好多人都在说什么安卓快凉了,工作越来越难找了。又是说什么程序员中年危机啥的,为啥我这年近30的老农根本没有这种感觉,反倒觉得那些贩卖焦虑的都是瞎j8扯谈。当然,职业危机意识确实是要有的,但根本没到那种草木皆兵的地步好吗?

                            Android凉了都是弱者的借口和说辞。虽然 Android 没有前几年火热了,已经过去了会四大组件就能找到高薪职位的时代了。这只能说明 Android 中级以下的岗位饱和了,现在高级工程师还是比较缺少的,很多高级职位给的薪资真的特别高(钱多也不一定能找到合适的),所以努力让自己成为高级工程师才是最重要的。

                            所以,最后这里放上我耗时两个月,将自己8年Android开发的知识笔记整理成的Android开发者必知必会系统学习资料笔记,上述知识点在笔记中都有详细的解读,里面还包含了腾讯、字节跳动、阿里、百度2019-2021面试真题解析,并且把每个技术点整理成了视频和PDF(知识脉络 + 诸多细节)。

                            [外链图片转存中…(img-vwFazyEh-1711730578956)]

                            以上全套学习笔记面试宝典,吃透一半保你可以吊打面试官,只有自己真正强大了,有核心竞争力,你才有拒绝offer的权力,所以,奋斗吧!骚年们!千里之行,始于足下。种下一颗树最好的时间是十年前,其次,就是现在。

                            最后,赠与大家一句诗,共勉!

                            不驰于空想,不骛于虚声。不忘初心,方得始终。

                            本文已被CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录

微信扫一扫加客服

微信扫一扫加客服

点击启动AI问答
Draggable Icon