本文实例讲述了Android实现双模(CDMA/GSM)手机短信监听的方法。分享给大家供大家参考,具体如下:
一、问题分析:
最近在做一个通过短信远程启动应用的功能,要用到短信监听,代码如下:
import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.os.Bundle; import android.telephony.SmsMessage; import android.widget.Toast; public class SMSReceiver extends BroadcastReceiver{ /*当收到短信时,就会触发此方法*/ public void onReceive(Context context, Intent intent) { Bundle bundle = intent.getExtras(); if(bundle!=null && bundle.get("pdus")!=null){ Object[] pdus = (Object[]) bundle.get("pdus"); //得到由短信内容组成的数组对象 if(pdus!=null && pdus.length>0){ SmsMessage[] messages = new SmsMessage[pdus.length]; for(int i=0;i
实际应用时发现双模手机对接收到的短信处理时总是在SmsMessage.createFromPdu的地方出现异常,异常信息:
java.lang.OutOfMemoryError: array size too large
at com.android.internal.telephony.cdma.SmsMessage.parsePdu(SmsMessage.java:658)
at com.android.internal.telephony.cdma.SmsMessage.createFromPdu(SmsMessage.java:116)
at android.telephony.SmsMessage.createFromPdu(SmsMessage.java:162)
而在android的源码中可以看到createFromPdu方法:
import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.database.ContentObserver; import android.database.Cursor; import android.net.Uri; import android.os.Handler; public class SMSReceiver extends BroadcastReceiver { private static final String SMS_RECEIVED = "android.provider.Telephony.SMS_RECEIVED"; private Context m_Context; private SmsContentObserver m_Smsobserver = new SmsContentObserver(new Handler()); @Override public void onReceive(Context context, Intent intent) { this.m_COntext= context; if (intent.getAction().equals(SMS_RECEIVED)) { //注册短信变化监听 context.getContentResolver().registerContentObserver(Uri.parse("content://sms/"), true, m_Smsobserver); } } /** * 短信内容观察者 * @author sinber * */ private class SmsContentObserver extends ContentObserver{ public SmsContentObserver(Handler handler) { super(handler); } /** * @Description 当短信表发送改变时,调用该方法 * 需要两种权限 *
如果是双模手机,调用此方法时会产生错误,问题就在于源码的TelephonyManager.getDefault().getPhoneType();该方法的返回值没有对应的双模手机的类型,而原生的android系统是不支持双模手机的。
二、解决办法:
我们可以采用广播接收者和内容观察者相结合的方式,直接读取手机的短信数据库,这样就避免了错误的产生,废话就不多说了,直接上代码:
import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.database.ContentObserver; import android.database.Cursor; import android.net.Uri; import android.os.Handler; public class SMSReceiver extends BroadcastReceiver { private static final String SMS_RECEIVED = "android.provider.Telephony.SMS_RECEIVED"; private Context m_Context; private SmsContentObserver m_Smsobserver = new SmsContentObserver(new Handler()); @Override public void onReceive(Context context, Intent intent) { this.m_COntext= context; if (intent.getAction().equals(SMS_RECEIVED)) { //注册短信变化监听 context.getContentResolver().registerContentObserver(Uri.parse("content://sms/"), true, m_Smsobserver); } } /** * 短信内容观察者 * @author sinber * */ private class SmsContentObserver extends ContentObserver{ public SmsContentObserver(Handler handler) { super(handler); } /** * @Description 当短信表发送改变时,调用该方法 * 需要两种权限 *
最后别忘了在AndroidManifest.xml中添加相应的权限,
还有别忘了注册广播接收者:
这样就能适应所有的android手机了,无论是双模还是单模都没问题,问题解决了。
更多关于Android相关内容感兴趣的读者可查看本站专题:《Android数据库操作技巧总结》、《Android编程之activity操作技巧总结》、《Android文件操作技巧汇总》、《Android编程开发之SD卡操作方法汇总》、《Android开发入门与进阶教程》、《Android资源操作技巧汇总》、《Android视图View技巧总结》及《Android控件用法总结》
希望本文所述对大家Android程序设计有所帮助。