????手機(jī)信號(hào)強(qiáng)度首先分為兩部分推姻,一個(gè)是WiFi信號(hào)強(qiáng)度平匈,一個(gè)是蜂巢網(wǎng)絡(luò)(數(shù)據(jù)流量)信號(hào)強(qiáng)度。
????信號(hào)強(qiáng)度會(huì)返回一個(gè)dbm單位的數(shù)據(jù)藏古,這個(gè)數(shù)據(jù)就代表了當(dāng)前環(huán)境下手機(jī)的信號(hào)如何增炭。具體信號(hào)強(qiáng)度等級(jí)請(qǐng)參考下方表格。
WiFi
范圍 | 等級(jí) |
---|---|
0 -(-55) | 信號(hào)很強(qiáng) |
(-55) - (-100) | 信號(hào)一般 |
<font color=red>信號(hào)強(qiáng)度拧晕,都是dbm為負(fù)值則正確隙姿,否則數(shù)據(jù)異常,越接近0信號(hào)越好</font>
????這些等級(jí)可以根據(jù)我們自己的需求來定義厂捞,信號(hào)范圍是(-100) - 0,我們可以確定自己需要輸出幾個(gè)等級(jí)输玷,從而調(diào)用方法WiFiManager#calculateSignalLevel()來推算出當(dāng)前WIFi信號(hào)強(qiáng)度屬于哪個(gè)等級(jí)。
????WiFi信號(hào)強(qiáng)度比較容易獲取靡馁,直接通過WifiManager對(duì)象來獲取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已經(jīng)幾乎沒人用了,所以沒有必要去獲取3G的)
范圍 | 等級(jí) |
---|---|
(-89)-(-51) | 滿格(Great) |
(-97)-(-89) | 很好(Good) |
(-103)-(-97) | 良好(MODERATE) |
(-107)- (-103) | 很差(poor) |
(-113) - (-107) | 無信號(hào)(min) |
????獲取4G的信號(hào)強(qiáng)度臭墨,還是有很多方法的貌虾,網(wǎng)上搜了一大堆,但是真正能用的還是下方我整合后的這種裙犹。因?yàn)?G手機(jī)現(xiàn)在市面上太多人用了尽狠,所以順便把5G獲取信號(hào)強(qiáng)度的方法也整合了進(jìn)去。
????先創(chuàng)建一個(gè)類繼承于PhoneStateListener叶圃,用于重寫監(jiān)聽方法袄膏,同時(shí)實(shí)現(xiàn)監(jiān)聽和取消監(jiān)聽方法以便我們工具調(diào)用。
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);
}
/**
* 開始監(jiān)聽
*/
public void register(){
if (tm == null){
callback.onGetMobileInfoResult(1);
return;
}
tm.listen(this,PhoneStateListener.LISTEN_SIGNAL_STRENGTHS);
}
/**
* 取消監(jiān)聽
*/
public void unRegister(){
if (tm != null){
tm.listen(this,PhoneStateListener.LISTEN_NONE);
}
}
@Override
public void onSignalStrengthsChanged(SignalStrength signalStrength) {
super.onSignalStrengthsChanged(signalStrength);
// 檢查權(quán)限
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來監(jiān)聽信號(hào)強(qiáng)度,當(dāng)信號(hào)強(qiáng)度刷新時(shí)德崭,觸發(fā)方法斥黑,生成一系列相關(guān)的Cell對(duì)象。我們獲取到所有CellInfo對(duì)象眉厨,然后取其中數(shù)值最小的dbm即可锌奴,或者也可以取這些dbm的平均值。
<font color=red>注意:這個(gè)方法會(huì)定時(shí)回調(diào)一次憾股,所以當(dāng)我們獲取到我們想要的數(shù)據(jù)后鹿蜀,需要調(diào)用tm.listen(phoneStateListener, PhoneStateListener.LISTEN_NONE);來取消注冊(cè)</font>
????對(duì)于5G,通過查看源碼發(fā)現(xiàn)其實(shí)就是多了一個(gè)實(shí)例對(duì)象服球,其他獲取方法都一樣茴恰。
關(guān)于RSSI
????<font color=red>Rssi意思就是信號(hào)強(qiáng)度的意思,與此有關(guān)系的還有一個(gè)RSRP斩熊,RSRP代表信號(hào)功率往枣。</font>
????關(guān)于兩者的詳細(xì)含義,具體看百度百科粉渠。
RSRP(Reference Signal Receiving Power)是在某個(gè)Symbol內(nèi)承載Reference Signal的所有RE上接收到的信號(hào)功百率的平均值;
RSSI(Received Signal Strength Indicator)則是在這個(gè)Symbol內(nèi)接收到的所有信號(hào)(包括度導(dǎo)頻信號(hào)和數(shù)據(jù)信號(hào),鄰區(qū)干擾信號(hào),噪音信號(hào)等)功率的平均值,是指接收的信號(hào)強(qiáng)度指示分冈,是無問限發(fā)送層的可選用部分,用來判定鏈接的質(zhì)量渣叛,以及是否增大廣播發(fā)送強(qiáng)答度;
RSRQ(Reference Signal Receiving Quality)則是RSRP和RSSI的比值,當(dāng)然因回為兩者測(cè)量所基于的帶寬可能不同,會(huì)用一個(gè)系數(shù)來調(diào)整,也就是 RSRQ = N*RSRP/RSS