Android 监听双卡信号强度

监听单卡信号强度

监听单卡的信号强度非常简单直接用TelephonyManager.listen()去监听sim卡的信号强度.
TelephonyManager = mTelephonyManager = (TelephonyManager) mContext.getSystemService(Context.TELEPHONY_SERVICE)
mTelephonyManager.listen(new PhoneStateListener(), PhoneStateListener.LISTEN_SIGNAL_STRENGTHS);

单这只是针对单卡的时候, 现在手机基本标配双卡, 如果要监测sim卡的信号强度就要考虑到双卡, 然而百度了一下, 并没有博文去介绍怎么去监测双卡, 这下只能靠自己了.

监测sim卡1的信号强度

//SubscriptionManager  该类主要包含了所有sim卡的信息
SubscriptionManager mSubscriptionManager = SubscriptionManager.from(mContext);
//获取sim卡1的信息
SubscriptionInfo sub0 = mSubscriptionManager.getActiveSubscriptionInfoForSimSlotIndex(0); if(null != sub0) {
//如果不为空 说明sim卡1存在
/*
sub0.getSubscriptionId() 获取sim卡1的 subId
*/
Sim1SignalStrengthsListener mSim1SignalStrengthsListener = new Sim1SignalStrengthsListener(sub0.getSubscriptionId());
//开始监听
mTelephonyManager.listen(mSim1SignalStrengthsListener, PhoneStateListener.LISTEN_SIGNAL_STRENGTHS);
}

上面只是设置并开始监听, 具体去监听哪一个sim卡的, 还需要具体去看 Sim1SignalStrengthsListener 中对 subId 的具体处理

PhoneStateListener 类中有一个成员变量 mSubId , 这个值就是当前默认操作的sim卡, 如果不明确指定, 默认值就是 SubscriptionManager.DEFAULT_SUBSCRIPTION_ID, 看PhoneStateListener在有带参构造设置subId, 但是都是hide隐藏, 既然无法初始化去设置subId, 那么我们就用Java强大的功能”反射”去设置mSubId的值, 设置完成之后, PhoneStateListener 就成了我们指定监测的sim卡了.

/*
* Subscription used to listen to the phone state changes
* @hide
*/
/** @hide */
protected int mSubId = SubscriptionManager.INVALID_SUBSCRIPTION_ID; /**
* Create a PhoneStateListener for the Phone with the default subscription
* using a particular non-null Looper.
* @hide
*/
public PhoneStateListener(Looper looper) {
this(SubscriptionManager.DEFAULT_SUBSCRIPTION_ID, looper);
}
/**
* Create a PhoneStateListener for the Phone using the specified subscription
* and non-null Looper.
* @hide
*/
public PhoneStateListener(int subId, Looper looper) {
...
}
 public static final int DEFAULT_SUBSCRIPTION_ID = Integer.MAX_VALUE;
  • 1
class Sim1SignalStrengthsListener extends PhoneStateListener {

        public Sim1SignalStrengthsListener(int subId) {
super();
ReflectUtil.setFieldValue(this, "mSubId", subId);
} @Override
public void onSignalStrengthsChanged(SignalStrength signalStrength) {
super.onSignalStrengthsChanged(signalStrength);
int level = getSignalStrengthsLevel(signalStrength);
}
}

反射代码 ReflectUtil



public static void setFieldValue(final Object obj, final String fieldName, final Object value) {
Field field = getAccessibleField(obj, fieldName); if (field == null) {
throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + obj + "]");
} try {
field.set(obj, value);
} catch (IllegalAccessException e) {
e.printStackTrace();
}
} public static Field getAccessibleField(final Object obj, final String fieldName) {
if (obj == null) {
throw new IllegalArgumentException("object can't be null");
} if (fieldName == null || fieldName.length() <= 0) {
throw new IllegalArgumentException("fieldName can't be blank");
} for (Class<?> superClass = obj.getClass(); superClass != Object.class; superClass = superClass.getSuperclass()) {
try {
Field field = superClass.getDeclaredField(fieldName);
makeAccessible(field);
return field;
} catch (NoSuchFieldException e) {
continue;
}
}
return null;
} public static void makeAccessible(Field field) {
if ((!Modifier.isPublic(field.getModifiers()) ||!Modifier.isPublic(field.getDeclaringClass().getModifiers()) || Modifier.isFinal(field.getModifiers())) && !field.isAccessible()) {
field.setAccessible(true);
}
}

监测sim卡2的信号强度

获取sim卡2的信息, 其他同sim卡1

...
SubscriptionInfo sub0 = mSubscriptionManager.getActiveSubscriptionInfoForSimSlotIndex(1);
...
  • 1

附完整代码

/**
* 手机信号监控(双卡)
* Created by zengyan on 2016/6/29.
*/
public class SignalStrengthsHandler { public static final String TAG = "SignalStrengthsManager";
public static final int INDEX_SIM1 = 0;
public static final int INDEX_SIM2 = 1;
private static SignalStrengthsHandler mInstance = null;
public static byte[] mLock = new byte[0];
private final Context mContext;
private final TelephonyManager mTelephonyManager;
private final SubscriptionManager mSubscriptionManager;
private final SimStateReceive mSimStateReceiver; private SimSignalInfo mSim1SignalInfo = new SimSignalInfo();
private SimSignalInfo mSim2SignalInfo = new SimSignalInfo(); private ArrayList<OnSignalStrengthsChangedListener> mOnSignalStrengthsChangedListeners = null;
private Sim1SignalStrengthsListener mSim1SignalStrengthsListener;
private Sim2SignalStrengthsListener mSim2SignalStrengthsListener; @TargetApi(Build.VERSION_CODES.LOLLIPOP_MR1)
private SignalStrengthsHandler() {
mSubscriptionManager = SubscriptionManager.from(mContext);
mTelephonyManager = (TelephonyManager) mContext.getSystemService(Context.TELEPHONY_SERVICE);
initListeners(); mSimStateReceiver = new SimStateReceive();
IntentFilter intentFilter = new IntentFilter();
intentFilter.addAction(SimStateReceive.ACTION_SIM_STATE_CHANGED);
mContext.registerReceiver(mSimStateReceiver, intentFilter);
} public static SignalStrengthsHandler getInstance() {
if (null == mInstance) {
synchronized (mLock) {
if (null == mInstance) {
mInstance = new SignalStrengthsHandler();
}
}
}
return mInstance;
} public void destroyInstance() {
if (null != mInstance) {
synchronized (mLock) {
if (null != mInstance) {
if (null != mOnSignalStrengthsChangedListeners) {
mOnSignalStrengthsChangedListeners.clear();
mOnSignalStrengthsChangedListeners = null;
}
mContext.unregisterReceiver(mSimStateReceiver);
mInstance = null;
}
}
}
} private void initListeners() {
listenSimSignalStrengths(SimCard.SIM_CARD_1);
listenSimSignalStrengths(SimCard.SIM_CARD_2);
} @TargetApi(Build.VERSION_CODES.LOLLIPOP_MR1)
private void listenSimSignalStrengths(SimCard simCard) {
if (simCard == SimCard.SIM_CARD_1) {
SubscriptionInfo sub0 = mSubscriptionManager.getActiveSubscriptionInfoForSimSlotIndex(INDEX_SIM1);
if (sub0 != null && null == mSim1SignalStrengthsListener) {
mSim1SignalStrengthsListener = new Sim1SignalStrengthsListener(sub0.getSubscriptionId());
}
mTelephonyManager.listen(mSim1SignalStrengthsListener, PhoneStateListener.LISTEN_SIGNAL_STRENGTHS);
} else if (simCard == SimCard.SIM_CARD_2) {
SubscriptionInfo sub1 = mSubscriptionManager.getActiveSubscriptionInfoForSimSlotIndex(INDEX_SIM2);
if (sub1 != null && null == mSim2SignalStrengthsListener) {
mSim2SignalStrengthsListener = new Sim2SignalStrengthsListener(sub1.getSubscriptionId());
}
mTelephonyManager.listen(mSim2SignalStrengthsListener, PhoneStateListener.LISTEN_SIGNAL_STRENGTHS);
}
} private void unListenSimSignalStrengths(SimCard simCard) {
if (simCard == SimCard.SIM_CARD_1) {
mSim1SignalInfo.mIsActive = false;
mSim1SignalInfo.mLevel = 0;
if (null != mSim1SignalStrengthsListener) {
mTelephonyManager.listen(mSim1SignalStrengthsListener, PhoneStateListener.LISTEN_NONE);
}
} else if (simCard == SimCard.SIM_CARD_2) {
mSim2SignalInfo.mIsActive = false;
mSim2SignalInfo.mLevel = 0;
if (null != mSim2SignalStrengthsListener) {
mTelephonyManager.listen(mSim2SignalStrengthsListener, PhoneStateListener.LISTEN_NONE);
}
} } /**
* 添加监听sim卡信号强度
*
* @param listener
*/
public void registerOnSignalStrengthsChangedListener(OnSignalStrengthsChangedListener listener) {
if (null == mOnSignalStrengthsChangedListeners) {
mOnSignalStrengthsChangedListeners = new ArrayList<>();
} if (mOnSignalStrengthsChangedListeners.contains(listener)) {
return;
} if (null != listener) {
mOnSignalStrengthsChangedListeners.add(listener);
}
} public void unregisterOnSignalStrengthsChangedListener(OnSignalStrengthsChangedListener listener) {
if (null == mOnSignalStrengthsChangedListeners) {
return;
} if (null == listener) {
return;
} if (mOnSignalStrengthsChangedListeners.contains(listener)) {
mOnSignalStrengthsChangedListeners.remove(listener);
}
} public void notyfyStateChange(boolean isSim1Exist, boolean isSim2Exist) {
if (null != mOnSignalStrengthsChangedListeners && !mOnSignalStrengthsChangedListeners.isEmpty()) {
for (int i = 0; i < mOnSignalStrengthsChangedListeners.size(); i++) {
OnSignalStrengthsChangedListener listener = mOnSignalStrengthsChangedListeners.get(i);
if (null != listener) {
listener.onSimStateChanged(isSim1Exist, isSim2Exist);
}
}
}
} public void notifyChange(SimCard simCard, int level) {
if (null != mOnSignalStrengthsChangedListeners && !mOnSignalStrengthsChangedListeners.isEmpty()) {
for (int i = 0; i < mOnSignalStrengthsChangedListeners.size(); i++) {
OnSignalStrengthsChangedListener listener = mOnSignalStrengthsChangedListeners.get(i);
if (null != listener) {
listener.onSignalStrengthsChanged(simCard, level);
}
}
}
} public boolean isSimCardExist(int cardIndex) {
boolean isSimCardExist = false;
try {
Method method = TelephonyManager.class.getMethod("getSimState", new Class[]{int.class});
int simState = (Integer) method.invoke(mTelephonyManager, new Object[]{Integer.valueOf(cardIndex)});
if (TelephonyManager.SIM_STATE_READY == simState) {
isSimCardExist = true;
}
} catch (Exception e) {
e.printStackTrace();
} return isSimCardExist;
} /**
* 获取sim信号 状态信息
*
* @return int[] index: 0:sim1 1:sim2
*/
public SimSignalInfo[] getSimSignalInfos() {
return new SimSignalInfo[]{mSim1SignalInfo, mSim2SignalInfo};
} private int getSignalStrengthsLevel(SignalStrength signalStrength) {
int level = -1;
try {
Method levelMethod = SignalStrength.class.getDeclaredMethod("getLevel");
level = (int) levelMethod.invoke(signalStrength);
} catch (Exception e) {
LogUtil.e(TAG, e.getMessage());
}
return level;
} private class Sim1SignalStrengthsListener extends PhoneStateListener { public Sim1SignalStrengthsListener(int subId) {
super();
//设置当前监听的sim卡
ReflectUtil.setFieldValue(this, "mSubId", subId);
} @Override
public void onSignalStrengthsChanged(SignalStrength signalStrength) {
super.onSignalStrengthsChanged(signalStrength);
int level = getSignalStrengthsLevel(signalStrength);
if (mSim1SignalInfo.mLevel == level) {
return;
}
mSim1SignalInfo.mLevel = level;
SignalStrengthsHandler.this.notifyChange(SimCard.SIM_CARD_1, mSim1SignalInfo.mLevel);
LogUtil.d(TAG, "sim 1 signal strengths level = " + mSim1SignalInfo.mLevel);
}
} private class Sim2SignalStrengthsListener extends PhoneStateListener { public Sim2SignalStrengthsListener(int subId) {
super();
//设置当前监听的sim卡
ReflectUtil.setFieldValue(this, "mSubId", subId);
} @Override
public void onSignalStrengthsChanged(SignalStrength signalStrength) {
super.onSignalStrengthsChanged(signalStrength);
int level = getSignalStrengthsLevel(signalStrength);
if (mSim2SignalInfo.mLevel == level) {
return;
}
mSim2SignalInfo.mLevel = level;
SignalStrengthsHandler.this.notifyChange(SimCard.SIM_CARD_2, mSim2SignalInfo.mLevel);
LogUtil.d(TAG, "sim 2 signal strengths level = " + mSim2SignalInfo.mLevel);
}
} public interface OnSignalStrengthsChangedListener {
void onSignalStrengthsChanged(SimCard simCard, int level); void onSimStateChanged(boolean isSim1Exist, boolean isSim2Exist);
} public enum SimCard {
SIM_CARD_1, SIM_CARD_2
} @TargetApi(Build.VERSION_CODES.LOLLIPOP_MR1)
class SimStateReceive extends BroadcastReceiver {
private final static String ACTION_SIM_STATE_CHANGED = "android.intent.action.SIM_STATE_CHANGED"; @Override
public void onReceive(Context context, Intent intent) {
LogUtil.i("SimStateReceive", "sim state changed");
if (intent.getAction().equals(ACTION_SIM_STATE_CHANGED)) {
mSim1SignalInfo.mIsActive = isSimCardExist(INDEX_SIM1)
&& null != mSubscriptionManager.getActiveSubscriptionInfoForSimSlotIndex(INDEX_SIM1);
mSim2SignalInfo.mIsActive = isSimCardExist(INDEX_SIM2)
&& null != mSubscriptionManager.getActiveSubscriptionInfoForSimSlotIndex(INDEX_SIM2); mSim1SignalInfo.mLevel = 0;
mSim2SignalInfo.mLevel = 0;
if (mSim1SignalInfo.mIsActive) {
listenSimSignalStrengths(SimCard.SIM_CARD_1);
} else {
unListenSimSignalStrengths(SimCard.SIM_CARD_1);
}
if (mSim2SignalInfo.mIsActive) {
listenSimSignalStrengths(SimCard.SIM_CARD_2);
} else {
unListenSimSignalStrengths(SimCard.SIM_CARD_2);
}
notyfyStateChange(mSim1SignalInfo.mIsActive, mSim2SignalInfo.mIsActive);
}
}
} public class SimSignalInfo {
/**
* 信号强度 0 - 5
*/
public int mLevel; /**
* sim卡是否有效
*/
public boolean mIsActive;
}
}
  • 1
 

Android 监听双卡信号强度(附完整代码)的更多相关文章

  1. Android监听系统短信数据库变化-提取短信内容

    由于监听系统短信广播受到权限的限制,所以很多手机可能使用这种方式没法监听广播,从而没办法获取到系统短信,所以又重新开辟一条路. Android监听系统短信数据库内容变化使用场景: 1.监听短信数据库的 ...

  2. Android监听应用程序安装和卸载

    Android监听应用程序安装和卸载 第一. 新建监听类:BootReceiver继承BroadcastReceiver package com.rongfzh.yc; import android. ...

  3. 【Android】Android 监听apk安装替换卸载广播

    [Android]Android 监听apk安装替换卸载广播 首先是要获取应用的安装状态,通过广播的形式 以下是和应用程序相关的Broadcast Action ACTION_PACKAGE_ADDE ...

  4. Android 监听 ScrollView 滑动到最底部。

    做产品时,有一个需求,需要监听ScrollView滑动到最底部.在网上找了些方法,都有这样或那样的问题,要不就是监听不精确, 要不就是重复监听,那些代码没有产品化,很不可靠. 经过自己试验,终于找到了 ...

  5. Android监听返回键、Home键+再按一次返回键退出应用

    Android监听返回键需重写onKeyDown()方法 Home键keyCode==KeyEvent.KEYCODE_HOME @Override public boolean onKeyDown( ...

  6. Android监听来电和去电

    要监听android打电话和接电话,只需下面2步骤1.第一步,写一个Receiver继承自BroadcastReceiver import android.app.Service; import an ...

  7. Android监听ScrollView滑动到顶端和底部

    Android监听ScrollView滑动到顶端和底部     package cn.testscrollview; import android.os.Bundle; import android. ...

  8. Android 监听网络变化

    Android 监听网络变化

  9. android 监听返回键

    android监听返回键 public boolean onKeyDown(int keyCode, KeyEvent event) { if (keyCode == KeyEvent.KEYCODE ...

随机推荐

  1. 设计模式之第6章-迭代器模式(Java实现)

    设计模式之第6章-迭代器模式(Java实现) “我已经过时了,就不要讲了吧,现在java自带有迭代器,还有什么好讲的呢?”“虽然已经有了,但是具体细节呢?知道实现机理岂不美哉?”“好吧好吧.”(迭代器 ...

  2. leetcode 【 Merge k Sorted Lists 】python 实现

    题目: Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexit ...

  3. leetcode 【 Merge Sorted Array 】python 实现

    题目: Given two sorted integer arrays A and B, merge B into A as one sorted array. Note:You may assume ...

  4. leetcode 【 Remove Element 】python 实现

    题目: Given an array and a value, remove all instances of that value in place and return the new lengt ...

  5. apizza导出为html后,从中提取api_name/api_path/api_method,保存到本地,方便根据接口名称得到接口路径与请求方法

    import re import os def open_file(file='c:/newcrm.html'): f=open(file,'r',encoding='utf-8') return f ...

  6. html基础标签之head和body标签

    什么是标签标签是由一对尖括号包裹的单词构成的,也有一些单闭和标签,仅仅就自己出现就可以了,例如meta,link 1. 这里介绍了几种html语言里常见的head标签 <!DOCTYPE htm ...

  7. 基于 Spring 和 iBATIS 的动态可更新多数据源持久层

    前言 我们时常会遇到一些 web 项目,需要从不同的数据源中抓取数据来进行分析,而这些数据源是有可能变化的,需要用户来进行动态的维护和添加.可是,大多数的 web 程序使用了应用服务器或者容器中间件来 ...

  8. hdu 4289 网络流拆点,类似最小割(可做模板)邻接矩阵实现

    Control Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Sub ...

  9. java _tomcat_mysql 部署

    项目做完了,要发布了,而Java的特长之一就是移植性好,面对着微软的XP的停止服务,Windows系统的“独裁”,越来越多的商家选择了开源的免费的linux系统作为服务器.因为linux系统也有图形界 ...

  10. POJ 3621 Sightseeing Cows(最优比例环+SPFA检测)

    Sightseeing Cows Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 10306   Accepted: 3519 ...