手機信号強度首先分為兩部分,一個是WiFi信号強度,一個是蜂巢網絡(資料流量)信号強度。
信号強度會傳回一個dbm機關的資料,這個資料就代表了目前環境下手機的信号如何。具體信号強度等級請參考下方表格。
WiFi
範圍 | 等級 |
---|---|
0 -(-55) | 信号很強 |
(-55) - (-100) | 信号一般 |
信号強度,都是dbm為負值則正确,否則資料異常,越接近0信号越好
這些等級可以根據我們自己的需求來定義,信号範圍是(-100) - 0,我們可以确定自己需要輸出幾個等級,進而調用方法WiFiManager#calculateSignalLevel()來推算出目前WIFi信号強度屬于哪個等級。
WiFi信号強度比較容易擷取,直接通過WifiManager對象來擷取Rssi即可。代碼如下:
WifiManager wifiManager = (WifiManager) context.getApplicationContext().getSystemService(Context.WIFI_SERVICE);
if (wifiManager != null) {
WifiInfo info = wifiManager.getConnectionInfo();
if (info != null) {
return info.getRssi();
}
}
4G/5G
(由于3G已經幾乎沒人用了,是以沒有必要去擷取3G的)
範圍 | 等級 |
---|---|
(-89)-(-51) | 滿格(Great) |
(-97)-(-89) | 很好(Good) |
(-103)-(-97) | 良好(MODERATE) |
(-107)- (-103) | 很差(poor) |
(-113) - (-107) | 無信号(min) |
擷取4G的信号強度,還是有很多方法的,網上搜了一大堆,但是真正能用的還是下方我整合後的這種。因為5G手機現在市面上太多人用了,是以順便把5G擷取信号強度的方法也整合了進去。
先建立一個類繼承于PhoneStateListener,用于重寫監聽方法,同時實作監聽和取消監聽方法以便我們工具調用。
public static class MyPhoneStateListener extends PhoneStateListener{
private InfoCallback callback;
private Context context;
private TelephonyManager tm;
public void init(@NonNull Context context,@NonNull InfoCallback callback){
this.context = context;
this.callback = callback;
this.tm = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE);
}
/**
* 開始監聽
*/
public void register(){
if (tm == null){
callback.onGetMobileInfoResult(1);
return;
}
tm.listen(this,PhoneStateListener.LISTEN_SIGNAL_STRENGTHS);
}
/**
* 取消監聽
*/
public void unRegister(){
if (tm != null){
tm.listen(this,PhoneStateListener.LISTEN_NONE);
}
}
@Override
public void onSignalStrengthsChanged(SignalStrength signalStrength) {
super.onSignalStrengthsChanged(signalStrength);
// 檢查權限
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
if (context.checkSelfPermission(Manifest.permission.ACCESS_COARSE_LOCATION)
!= PackageManager.PERMISSION_GRANTED) {
callback.onGetMobileInfoResult(2);
return;
}
}
List<CellInfo> cellInfoList = tm.getAllCellInfo();
List<Integer> dbms = new ArrayList<>();
if (null != cellInfoList) {
for (CellInfo cellInfo : cellInfoList) {
try {
int dbm = 1;
if (cellInfo instanceof CellInfoGsm) {
CellSignalStrengthGsm cellSignalStrengthGsm = ((CellInfoGsm) cellInfo).getCellSignalStrength();
dbm = cellSignalStrengthGsm.getDbm();
} else if (cellInfo instanceof CellInfoCdma) {
CellSignalStrengthCdma cellSignalStrengthCdma = ((CellInfoCdma) cellInfo).getCellSignalStrength();
dbm = cellSignalStrengthCdma.getDbm();
} else if (cellInfo instanceof CellInfoWcdma) {
CellSignalStrengthWcdma cellSignalStrengthWcdma = ((CellInfoWcdma) cellInfo).getCellSignalStrength();
dbm = cellSignalStrengthWcdma.getDbm();
} else if (cellInfo instanceof CellInfoLte) {
CellSignalStrengthLte cellSignalStrengthLte = ((CellInfoLte) cellInfo).getCellSignalStrength();
dbm = cellSignalStrengthLte.getDbm();
} else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q && cellInfo instanceof CellInfoNr) { // 5G
CellSignalStrengthNr cellSignalStrengthNr = (CellSignalStrengthNr) ((CellInfoNr) cellInfo).getCellSignalStrength();
dbm = cellSignalStrengthNr.getDbm();
}
dbms.add(dbm);
} catch (Exception ignore) {
}
}
}
Collections.sort(dbms);
if (dbms.size() > 1) {
callback.onGetMobileInfoResult(dbms.get(dbms.size() - 1));
} else {
callback.onGetMobileInfoResult(1);
}
}
}
之後,我們可以直接封裝方法。
public static void getMobileSignalStrength(@NonNull final Context context,
@NonNull final InfoCallback callback) {
final TelephonyManager tm = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE);
if (tm == null) {
callback.onGetMobileInfoResult(1);
return ;
}
final MyPhoneStateListener myPhoneStateListener = new MyPhoneStateListener();
myPhoneStateListener.init(context, new GetMobileInfoCallback() {
@Override
public void onGetMobileInfoResult(int rssi) {
myPhoneStateListener.unRegister();
callback.onGetMobileInfoResult(rssi);
}
});
myPhoneStateListener.register();
}
核心思想就是用PhoneStateListener來監聽信号強度,當信号強度重新整理時,觸發方法,生成一系列相關的Cell對象。我們擷取到所有CellInfo對象,然後取其中數值最小的dbm即可,或者也可以取這些dbm的平均值。
注意:這個方法會定時回調一次,是以當我們擷取到我們想要的資料後,需要調用tm.listen(phoneStateListener, PhoneStateListener.LISTEN_NONE);來取消注冊
對于5G,通過檢視源碼發現其實就是多了一個執行個體對象,其他擷取方法都一樣。
關于RSSI
Rssi意思就是信号強度的意思,與此有關系的還有一個RSRP,RSRP代表信号功率。
關于兩者的詳細含義,具體看百度百科。
- RSRP(Reference Signal Receiving Power)是在某個Symbol内承載Reference Signal的所有RE上接收到的信号功百率的平均值;
- RSSI(Received Signal Strength Indicator)則是在這個Symbol内接收到的所有信号(包括度導頻信号和資料信号,鄰區幹擾信号,噪音信号等)功率的平均值,是指接收的信号強度訓示,是無問限發送層的可選用部分,用來判定連結的品質,以及是否增大廣播發送強答度;
- RSRQ(Reference Signal Receiving Quality)則是RSRP和RSSI的比值,當然因回為兩者測量所基于的帶寬可能不同,會用一個系數來調整,也就是 RSRQ = N*RSRP/RSS