欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android实现双模(CDMA/GSM)手机短信监听的方法

程序员文章站 2024-03-03 15:09:22
本文实例讲述了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程序设计有所帮助。