双模(CDMA/GSM)手机实现短信监听

来源:互联网 发布:sql默认值语句 编辑:程序博客网 时间:2024/04/28 14:03
一、问题分析:
最近在做一个通过短信远程启动应用的功能,要用到短信监听,代码如下:
  1. 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<pdus.length;i++){ byte[] pdu = (byte[]) pdus[i]; //得到短信内容,内容是以pdu格式存放的 messages[i] = SmsMessage.createFromPdu(pdu); } for(SmsMessage msg:messages){ String smscontent = msg.getMessageBody(); //得到短信内容 String smssender = msg.getOriginatingAddress(); //得到短信发送者的手机号 } } } }}
复制代码

实际应用时发现双模手机对接收到的短信处理时总是在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方法:
  1. public static SmsMessage createFromPdu(byte[] pdu) { SmsMessageBase wrappedMessage; int activePhone = TelephonyManager.getDefault().getPhoneType(); if (PHONE_TYPE_CDMA == activePhone) { wrappedMessage = com.android.internal.telephony.cdma.SmsMessage.createFromPdu(pdu); } else { wrappedMessage = com.android.internal.telephony.gsm.SmsMessage.createFromPdu(pdu); } return new SmsMessage(wrappedMessage);
复制代码

如果是双模手机,调用此方法时会产生错误,问题就在于源码的TelephonyManager.getDefault().getPhoneType();该方法的返回值没有对应的双模手机的类型,而原生的android系统是不支持双模手机的。

二、解决办法:

我们可以采用广播接收者和内容观察者相结合的方式,直接读取手机的短信数据库,这样就避免了错误的产生,废话就不多说了,直接上代码:
  1. import android.content.BroadcastReceiver;
  2. import android.content.Context;
  3. import android.content.Intent;
  4. import android.database.ContentObserver;
  5. import android.database.Cursor;
  6. import android.net.Uri;
  7. import android.os.Handler;
  8. public class SMSReceiver extends BroadcastReceiver {
  9. private static final String SMS_RECEIVED = "android.provider.Telephony.SMS_RECEIVED";
  10. private Context m_Context;
  11.  private SmsContentObserver m_Smsobserver = new SmsContentObserver(new Handler());
  12.  @Override
  13.  public void onReceive(Context context, Intent intent) {
  14. this.m_Context = context;
  15. if (intent.getAction().equals(SMS_RECEIVED)) {
  16. //注册短信变化监听
  17. context.getContentResolver().registerContentObserver(Uri.parse("content://sms/"), true, m_Smsobserver); }
  18.  }
  19. /** * 短信内容观察者 * @author sinber * */
  20. private class SmsContentObserver extends ContentObserver{
  21.  public SmsContentObserver(Handler handler) { super(handler); }
  22. /** * @Description 当短信表发送改变时,调用该方法 * 需要两种权限 * <li>android.permission.READ_SMS读取短信 </li> * <li>android.permission.WRITE_SMS写短信 </li> * @Author sinebr * */
  23. @Override
  24. public void onChange(boolean selfChange) {
  25. super.onChange(selfChange);
  26. Cursor cursor = null;
  27.  try{
  28. //读取收件箱中的短信
  29. cursor = m_Context.getContentResolver().query(Uri.parse("content://sms/inbox"), null, null, null, "date desc");
  30. String body; boolean hasDone = false;
  31.  if (cursor != null){
  32. while (cursor.moveToNext()){
  33. body = cursor.getString(cursor.getColumnIndex("body"));
  34.  if(body != null && body.equals("【startMyActivity】"))
  35. { //此处略去启动应用的代码
  36.  hasDone = true; break;
  37.  }
  38. if (hasDone){ break; } } } }
  39. catch(Exception e){ e.printStackTrace();
  40.  }
  41. finally{
  42. if(cursor!=null) cursor.close();
  43.  }
  44.  }
  45. }
  46. }
最后别忘了在AndroidManifest.xml中添加相应的权限,
  1. <!-- 接收短信权限 --> <uses-permission android:name="android.permission.RECEIVE_SMS"/> <!-- 发送短信权限 --> <uses-permission android:name="android.permission.SEND_SMS"/>
还有别忘了注册广播接收者
  1. <receiver android:name=".SMSReceiver"> <intent-filter> <action android:name="android.provider.Telephony.SMS_RECEIVED"/> </intent-filter></receiver>
复制代码

这样就能适应所有的android手机了,无论是双模还是单模都没问题,问题解决了。



0 0