Android实现双模(CDMA/GSM)手机短信监听的方法
本文实例讲述了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<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方法:
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 当短信表发送改变时,调用该方法 * 需要两种权限 *<li>android.permission.read_sms读取短信 </li> *<li>android.permission.write_sms写短信 </li> * @author sinebr * */ @override public void onchange(boolean selfchange) { super.onchange(selfchange); cursor cursor = null; try{ //读取收件箱中的短信 cursor = m_context.getcontentresolver().query(uri.parse("content://sms/inbox"), null, null, null, "date desc"); string body; boolean hasdone = false; if (cursor != null){ while (cursor.movetonext()){ body = cursor.getstring(cursor.getcolumnindex("body")); if(body != null && body.equals("【startmyactivity】")){ //此处略去启动应用的代码 hasdone = true; break; } if (hasdone){ break; } } } }catch(exception e){ e.printstacktrace(); }finally{ if(cursor!=null) cursor.close(); } } } }
如果是双模手机,调用此方法时会产生错误,问题就在于源码的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 当短信表发送改变时,调用该方法 * 需要两种权限 * <li>android.permission.read_sms读取短信 </li> * <li>android.permission.write_sms写短信 </li> * @author sinebr * */ @override public void onchange(boolean selfchange) { super.onchange(selfchange); cursor cursor = null; try{ //读取收件箱中的短信 cursor = m_context.getcontentresolver().query(uri.parse("content://sms/inbox"), null, null, null, "date desc"); string body; boolean hasdone = false; if (cursor != null){ while (cursor.movetonext()){ body = cursor.getstring(cursor.getcolumnindex("body")); if(body != null && body.equals("【startmyactivity】")){ //此处略去启动应用的代码 hasdone = true; break; } if (hasdone){ break; } } } }catch(exception e){ e.printstacktrace(); }finally{ if(cursor!=null) cursor.close(); } } } }
最后别忘了在androidmanifest.xml中添加相应的权限,
<!-- 接收短信权限 --> <uses-permission android:name="android.permission.receive_sms"/> <!-- 发送短信权限 --> <uses-permission android:name="android.permission.send_sms"/>
还有别忘了注册广播接收者:
<receiver android:name=".smsreceiver"> <intent-filter> <action android:name="android.provider.telephony.sms_received"/> </intent-filter> </receiver>
这样就能适应所有的android手机了,无论是双模还是单模都没问题,问题解决了。
更多关于android相关内容感兴趣的读者可查看本站专题:《android数据库操作技巧总结》、《android编程之activity操作技巧总结》、《android文件操作技巧汇总》、《android编程开发之sd卡操作方法汇总》、《android开发入门与进阶教程》、《android资源操作技巧汇总》、《android视图view技巧总结》及《android控件用法总结》
希望本文所述对大家android程序设计有所帮助。
下一篇: Python中的浮点数原理与运算分析
推荐阅读
-
Android实现双模(CDMA/GSM)手机短信监听的方法
-
android 获取手机GSM/CDMA信号信息,并获得基站信息的方法
-
Android监听输入法弹窗和关闭的实现方法
-
Android监听输入法弹窗和关闭的实现方法
-
全面解析Android中对EditText输入实现监听的方法
-
Android监听手机电话状态与发送邮件通知来电号码的方法(基于PhoneStateListene实现)
-
Android监听手机电话状态与发送邮件通知来电号码的方法(基于PhoneStateListene实现)
-
Android编程实现音量按钮添加监听事件的方法
-
Android编程实现EditText字数监听并显示的方法
-
Android编程使用GestureDetector实现简单手势监听与处理的方法