具体的替换流程:先用新进的短信生成一个ContentValues,再用短信的地址和协议标识当作条件到数据库中去查询,如果查到了,就替换,否则就存储。
存储的流程,也是先生成一个CotentValues,然后取出短信的Thread Id和地址,地址要与联系人数据库同步一下,以保证是能识别的地址。如果Thread Id不是合法的,那么就用同步过的地址尝试重新生成Thread Id,尝试5次。然后把刷新过的Thread Id放到ContentValues中,把ContentValues插入到数据库中。如果设置为把信息存储到SIM卡,还要调用SmsManager把信息拷贝到SIM卡上。计算短信的大小,并更新至数据库。删除过期的短信,和超过数量限制的短信,然后返回插入后得到的短信Uri。
最后,对于替换或插入的短信,用Uri去StatusBar做Notification。
GUI在刷新列表时也能得到新短信,因为短信已经被存储到数据库中。
短信接收的时序图如图所示:
三、双卡双待手机解析短信异常分析及解决
由于是双SIM卡,而且两个卡槽支持的运营商或者网络制式不一定相同,比如一个卡槽支持WCDMA,另一个却只支持GSM ,导致用正常方法解析短信很容易遇到异常。
这里先看下解决方案,这里需要以反射的方式解析不同类型的短信,并且对于不同机型,需对应地进行调整适配:
获取短信信息,注意:为解决双卡双待手机解析短信异常问题,使用Java反射机制,优先解析GSM类型的短信,假如解析失败才按CDMA类型的短信进行解析)
public static SmsMessage[] getSmsMessage(Intent intent) {
SmsMessage[] msgs = null;
Object messages[] = (Object[]) intent.getSerializableExtra("pdus");
int len = 0;
if (null != messages && (len = messages.length) > 0) {
msgs = new SmsMessage[len];
try {
for (int i = 0; i < len; i++) {
SmsMessage message = null;
if ("GSM".equals(intent.getStringExtra("from"))) { // 适配MOTO XT800双卡双待手机
message = createFromPduGsm((byte[]) messages[i]);
} else if ("CDMA".equals(intent.getStringExtra("from"))) { // 适配MOTO XT800双卡双待手机
message = createFromPduCdma((byte[]) messages[i]);
} else {
message = SmsMessage.createFromPdu((byte[]) messages[i]); // 系统默认的解析短信方式
}
if (null == message) { // 解决双卡双待类型手机解析短信异常问题
message = createFromPduGsm((byte[]) messages[i]);
if (null == message) {
message = createFromPduCdma((byte[]) messages[i]);
}
}
if (null != message) {
msgs[i] = message;
}
}
} catch (Exception e) {
e.printStackTrace();
msgs = getSmsMessageByReflect(intent); // 解决双卡双待手机解析短信异常问题
} catch (Error er) {
er.printStackTrace();
msgs = getSmsMessageByReflect(intent); // 解决双卡双待手机解析短信异常问题
}
}
return msgs;
}











