10分鐘讓你實如今APP中對網絡狀態變化進行全局提示

永遠不要指望用戶按照你預設的步驟操做 APP

一個新項目剛剛開始推廣工做,市場人員向我抱怨用戶使用時總會出現各類各樣的問題,大部分問題都是由於用戶操做不當致使的,可是在用戶眼中的結論就是「大家的 APP 很差用」。java

舉一個例子,有的用戶在使用時禁用了 APP 訪問移動網絡,或者有的用戶乾脆都沒有打開移動數據開關或者 WIFI 開關。可是做爲開發人員,咱們應該避免用戶思考,當用戶使用出現問題時,APP 應該可以引導用戶前往設置,故有此文。android

咱們但願當用戶網絡鏈接不可用時,及時提醒用戶當前的網絡狀態。當鏈接恢復時,將提示用的視圖隱藏,而且咱們但願這個提示視圖能夠工做在全部須要網絡的頁面中。面試

思路以下:使用 BaseActivity ,全部頁面繼承該文件,在該文件中實現根據網絡狀態顯示提示、隱藏提示。性能優化

好了,廢話少說,show u the code。網絡

1. 實現監聽網絡狀態變動的廣播接收器

咱們使用廣播接收器接收網絡變化的 Intent,這裏直接使用靜態註冊的方法,由於咱們不須要在每一個頁面單獨註冊這個 Receiver,那過重量級了。架構

NetworkConnectChangedReceiver.java
    
    public class NetworkConnectChangedReceiver extends BroadcastReceiver {
        private static final String TAG = "NetworkConnectChanged";
        @Override
        public void onReceive(Context context, Intent intent) {
            //**判斷當前的網絡鏈接狀態是否可用*/
            boolean isConnected = NetUtils.isConnected(context);
            Log.d(TAG, "onReceive: 當前網絡 " + isConnected);
            EventBus.getDefault().post(new NetworkChangeEvent(isConnected));
        }
    }

事件Event:ide

public class NetworkChangeEvent {
        public boolean isConnected; //是否存在網絡
    
        public NetworkChangeEvent(boolean isConnected) {
            this.isConnected = isConnected;
        }
    }

判斷網絡鏈接是否可用:佈局

/**
         * 判斷網絡是否鏈接
         * @param context
         * @return
         */
        public static boolean isConnected(Context context) {
            ConnectivityManager connectivity = (ConnectivityManager) context
                    .getSystemService(Context.CONNECTIVITY_SERVICE);
    
            if (null != connectivity) {
                NetworkInfo info = connectivity.getActiveNetworkInfo();
                if (null != info && info.isConnected()) {
                    if (info.getState() == NetworkInfo.State.CONNECTED) {
                        return true;
                    }
                }
            }
            return false;
        }

靜態註冊Receiver:post

<receiver android:name=".receiver.NetworkConnectChangedReceiver">
        <intent-filter>
            <action android:name="android.NET.conn.CONNECTIVITY_CHANGE" />
            <action android:name="android.Net.wifi.WIFI_STATE_CHANGED" />
            <action android:name="android.net.wifi.STATE_CHANGE" />
        </intent-filter>
    </receiver>

2. 在 BaseActivity中監聽事件並處理提示視圖

看到 EventBus 的時候你是否是已經知道個人實現方式了(笑 XD),是的就是那個已經好久沒人提了的 EventBus。固然還可使用觀察者模式來實現,這樣就不用依賴第三方庫了,可是咱們須要的是快速實現,且對原有代碼儘量少的改動,引入觀察者模式顯然不如直接拿 EventBus來的方便。性能

BaseActivity.java

public class BaseActivity extends Activity {
    
        protected Context mContext;
        protected ACache mACache;
        protected boolean mCheckNetWork = true; //默認檢查網絡狀態
        View mTipView;
        WindowManager mWindowManager;
        WindowManager.LayoutParams mLayoutParams;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            mContext = this;
            this.mACache = ACache.get(mContext);
            MyApp.addActivity(this);
            initTipView();//初始化提示View
            EventBus.getDefault().register(this);
        }
    
        @Override
        protected void onResume() {
            super.onResume();
            MobclickAgent.onResume(this);
            //在無網絡狀況下打開APP時,系統不會發送網絡情況變動的Intent,須要本身手動檢查
            hasNetWork(NetUtils.isConnected(mContext));
        }
    
        @Override
        protected void onPause() {
            super.onPause();
            MobclickAgent.onPause(this);
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            MyApp.removeActivity(this);
            EventBus.getDefault().unregister(this);
        }
    
        @Override
        public void finish() {
            super.finish();
            //當提示View被動態添加後直接關閉頁面會致使該View內存溢出,因此須要在finish時移除
            if (mTipView != null && mTipView.getParent() != null) {
                mWindowManager.removeView(mTipView);
            }
        }
    
        @Subscribe(threadMode = ThreadMode.MAIN)
        public void onNetworkChangeEvent(NetworkChangeEvent event) {
            hasNetWork(event.isConnected);
        }
    
        private void hasNetWork(boolean has) {
            if (isCheckNetWork()) {
                if (has) {
                    if (mTipView != null && mTipView.getParent() != null) {
                        mWindowManager.removeView(mTipView);
                    }
                } else {
                    if (mTipView.getParent() == null) {
                        mWindowManager.addView(mTipView, mLayoutParams);
                    }
                }
            }
        }
    
        public void setCheckNetWork(boolean checkNetWork) {
            mCheckNetWork = checkNetWork;
        }
    
        public boolean isCheckNetWork() {
            return mCheckNetWork;
        }
    
        private void initTipView() {
            LayoutInflater inflater = getLayoutInflater();
            mTipView = inflater.inflate(R.layout.layout_network_tip, null); //提示View佈局
            mWindowManager = (WindowManager) this.getSystemService(Context.WINDOW_SERVICE);
            mLayoutParams = new WindowManager.LayoutParams(
                    ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT,
                    WindowManager.LayoutParams.TYPE_APPLICATION,
                    WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
                            | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE,
                    PixelFormat.TRANSLUCENT);
            //使用非CENTER時,能夠經過設置XY的值來改變View的位置
            mLayoutParams.gravity = Gravity.TOP;
            mLayoutParams.x = 0;
            mLayoutParams.y = 0;
        }
    }

默認全部繼承 BaseActivity 的頁面當網絡情況變化活無網絡時都會顯示提示,若是某個頁面不須要網絡狀態提示,能夠在該頁面 onCreate 方法中調用 setCheckNetWork(false) 便可。

因爲我所有頁面都有一個50dp高度的 toolbar,因此我直接在 R.layout.layout_network_tip 文件中設置了上邊距。你也能夠在 BaseActivity 中經過方法來設置 mLayoutParams.x = 0;mLayoutParams.y = 0; 來使每一個頁面動態設置提示的位置。

最終效果以下圖:

在這裏插入圖片描述

ToDo

全部頁面在網絡連接恢復後應該能夠自動從新發起網絡請求,實現原理其實也很簡單,在BaseActivity中增長一個reConnect()的方法,在網絡恢復去除提示View的時候調用。在各個頁面中重寫該方法便可。

免費獲取安卓開發架構的資料(包括Fultter、高級UI、性能優化、架構師課程、 NDK、Kotlin、混合式開發(ReactNative+Weex)和一線互聯網公司關於android面試的題目彙總能夠加:936332305 / 連接:點擊連接加入【安卓開發架構】:https://jq.qq.com/?_wv=1027&k...

相關文章
相關標籤/搜索