- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章Android开发中使用mms模块收发单卡和双卡短信的教程由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
一.信息发送: com.android.mms.data.workingmessage.java 类 send()函数: 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
|
public
void
send() {
......
if
(requiresmms() || addresscontainsemailtomms(conv, msgtxt)) {
// 彩信
slideshow.prepareforsend();
new
thread(
new
runnable() {
public
void
run() {
sendmmsworker(conv, mmsuri, persister, slideshow, sendreq);
}
}).start();
}
else
{
// 短信
new
thread(
new
runnable() {
public
void
run() {
presendsmsworker(conv, msgtext);
}
}).start();
......
}
|
prapareforsend(). 先确保有slidshow,也就是实质内容。 确保文字已拷贝。确保标题。 根据消息分类,如果是短信直接起一个线程,跑presendsmsworker函数,发送短信;如果是彩信,先跑prapareforsave确保文本信息,然后起一个线程,单独跑sendmmsworker函数。不管是短信还是彩信,起了那个worker函数之一就算发送信息成功了。 最后修改recipient cache, 重置标志位,过程就结束了。 短信发送先调用presendsmsworker函数,在presendsmsworker函数中又起了sendsmsworker函数.
1
2
3
4
5
6
7
8
9
10
11
|
private
void
sendsmsworker(string msgtext, string semiseprecipients,
long
threadid) {
......
messagesender sender =
new
smsmessagesender(mcontext, dests, msgtext, threadid);
sender.sendmessage(threadid);
......
}
|
smsmessagesender.java类,在mms/transaction下面,实现了messagesender接口,这个接口sendmessage并返回boolean的值。若发送的是mms,返回true。若发送的是sms,返回false.
当然,对于单卡手机和双卡双待手机的短信发送流程是有区别的(短信接收的流程是相同的,相对流程也比较简洁),关于具体的流程还是直接用uml图来说明更为直接: 信息发送与接收时序图: 发送短信 单卡手机短信发送的时序图如图所示:
双卡手机短信发送的时序图则如下图所示:
二.短信的接收 信息的接收工作是由底层来完成的,当有一个 新的信息时底层完成接收后会以intent的方式来通知上层应用,信息的相关内容也包含在intent当中,android所支持的信息intent都定义在android.provider.telephony.intents里面。 短信接收,对于上层应用程序来讲就是要处理广播事件sms_received_action,它是由frameworks发出告诉上层有新的sms已收到。在mms中,是由privilegedsmsreceiver来处理,它收到sms_received_action(android.provider.telephony.intents.sms_received_action=”android.provider.telephony.sms_received”)后会启动smsreceiverservice来做具体的处理。 smsreceiverservice会先检查短信的类型,如果是class0短信,直接在gui中显示,不做任何其他的处理,也即不会存储到数据库中,也不会在notification bar中做notification。 对于其他短信,会进行替换现有的消息,或是当作新消息插入。原则就是如果在数据库中已有的短信中,与新来的短信的原始地址和协议标识都一样,那么就把其替换成新进的短信,否则就当作新短信插入。 具体的替换流程:先用新进的短信生成一个contentvalues,再用短信的地址和协议标识当作条件到数据库中去查询,如果查到了,就替换,否则就存储。 存储的流程,也是先生成一个cotentvalues,然后取出短信的thread id和地址,地址要与联系人数据库同步一下,以保证是能识别的地址。如果thread id不是合法的,那么就用同步过的地址尝试重新生成thread id,尝试5次。然后把刷新过的thread id放到contentvalues中,把contentvalues插入到数据库中。如果设置为把信息存储到sim卡,还要调用smsmanager把信息拷贝到sim卡上。计算短信的大小,并更新至数据库。删除过期的短信,和超过数量限制的短信,然后返回插入后得到的短信uri。 最后,对于替换或插入的短信,用uri去statusbar做notification。 gui在刷新列表时也能得到新短信,因为短信已经被存储到数据库中.
短信接收的时序图如图所示:
。
3、双卡双待手机解析短信异常分析及解决 由于是双sim卡,而且两个卡槽支持的运营商或者网络制式不一定相同,比如一个卡槽支持wcdma,另一个却只支持gsm ,导致用正常方法解析短信很容易遇到异常。 这里先看下解决方案,这里需要以反射的方式解析不同类型的短信,并且对于不同机型,需对应地进行调整适配
获取短信信息,注意:为解决双卡双待手机解析短信异常问题,使用java反射机制,优先解析gsm类型的短信,假如解析失败才按cdma类型的短信进行解析) 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
|
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;
}
|
反射方式获取短信 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
|
/**
* 使用java反射机制获取短信信息(解决双卡双待手机解析短信异常问题,优先解析gsm类型的短信,假如解析失败才按cdma类型的短信进行解析)
*
* @param intent
* @return
*/
private
static
smsmessage[] getsmsmessagebyreflect(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 = createfrompdugsm((
byte
[]) messages[i]);
if
(
null
== message) {
message = createfrompducdma((
byte
[]) messages[i]);
}
if
(
null
!= message) {
msgs[i] = message;
}
}
}
catch
(securityexception e) {
e.printstacktrace();
}
catch
(illegalargumentexception e) {
e.printstacktrace();
}
catch
(classnotfoundexception e) {
e.printstacktrace();
}
catch
(nosuchmethodexception e) {
e.printstacktrace();
}
catch
(illegalaccessexception e) {
e.printstacktrace();
}
catch
(invocationtargetexception e) {
e.printstacktrace();
}
catch
(instantiationexception e) {
e.printstacktrace();
}
}
return
msgs;
}
|
。
通过java反射机制解析gsm类型的短信
1
2
3
|
private
static
smsmessage createfrompdugsm(
byte
[] pdu)
throws
securityexception, illegalargumentexception, classnotfoundexception, nosuchmethodexception, illegalaccessexception, invocationtargetexception, instantiationexception {
return
createfrompdu(pdu,
"com.android.internal.telephony.gsm.smsmessage"
);
}
|
解析cdma类型的短信 。
1
2
3
|
private
static
smsmessage createfrompducdma(
byte
[] pdu)
throws
securityexception, illegalargumentexception, classnotfoundexception, nosuchmethodexception, illegalaccessexception, invocationtargetexception, instantiationexception {
return
createfrompdu(pdu,
"com.android.internal.telephony.cdma.smsmessage"
);
}
|
解析gsm或者cdma类型的短信 。
1
2
3
4
5
6
7
8
9
10
11
|
private
static
smsmessage createfrompdu(
byte
[] pdu, string classname)
throws
classnotfoundexception, securityexception, nosuchmethodexception, illegalargumentexception, illegalaccessexception, invocationtargetexception, instantiationexception {
class
<?> clazz =
class
.forname(classname);
object object = clazz.getmethod(
"createfrompdu"
,
byte
[].
class
).invoke(clazz.newinstance(), pdu);
if
(
null
!= object) {
constructor<?> constructor = smsmessage.
class
.getdeclaredconstructor(
class
.forname(
"com.android.internal.telephony.smsmessagebase"
));
constructor.setaccessible(
true
);
return
(smsmessage) constructor.newinstance(object);
}
else
{
return
null
;
}
}
|
最后此篇关于Android开发中使用mms模块收发单卡和双卡短信的教程的文章就讲到这里了,如果你想了解更多关于Android开发中使用mms模块收发单卡和双卡短信的教程的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
在 Android 的 API > 19 中是否有任何方法可以获取可移动 SD 卡的路径? 与外部 SD 卡一样,我们有 Environment.getExternalStorageDirectory
一些 Android 设备有 microSD(或其他存储卡)插槽,通常安装为 /storage/sdcard1 据我所知,自 Android 4.4 起 Google 限制了对此内存的访问,并在 An
我使用 Java Card 2.1.2 SDK 和 GPShell 作为与设备通信的方式在 Java Card 上构建一个项目。我从 GpShell 测试了 helloworld 示例,并成功发送了
我开发了一个应用程序,它有一个来电接收器,它适用于所有手机。一位用户有一部双 SIM 卡安卓手机。该应用程序适用于第一张 SIM 卡。但是当有人调用他的第二张 SIM 卡时,我们的应用程序不会被调用。
我有一个带预览的文件输入。 这是笔 Codepen 我想强制高度,我无法理解我该怎么做。我想将此组件的高度固定为 300px(示例),我还需要保持加载图像的正确纵横比,用灰色背景填充空白。现在我保持宽
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 想改进这个问题?将问题更新为 on-topic对于堆栈溢出。 6年前关闭。 Improve this qu
我正在使用此代码访问 SD card : import os from os.path import join from jnius import autoclass #from android.pe
我正在为数据记录设备编写固件。它以 20 Hz 的频率从传感器读取数据并将数据写入 SD 卡。但是,向SD卡写入数据的时间并不一致(大约200-300 ms)。因此,一种解决方案是以一致的速率将数据写
我正在使用以下代码将视频放到网站上,但是在垂直方向上,手机屏幕上只能看到视频的左半部分 我不是网络开发人员。有人可以告诉我确切的内容吗,如何使其正确放置在手机屏幕上? 是在youtube iframe
我正在使用 Vuetify 1.5 和 Vuetify 网格系统来设置我的布局。现在我有一个组件 HelloWorld我将其导入到我的 Parent 中成分。我已经在我的 HelloWorld 中设置
我使用 python 制作了一个简单的二十一点游戏。我制作了游戏的其余部分,但我正在努力放入 ASCII 卡,所以这只是代码的一小部分。我尝试将 * len(phand) 放在附加行的末尾。虽然这确实
我正在使用玩家卡设置 Twitter 卡。它可以在预览工具中运行,但文档说它需要在“twitter.com 现代桌面浏览器? native iOs 和 Android Twitter 应用程序?mob
任何旧的 GSM 兼容 SIM 卡(3G USIM 的奖励)。 我想我需要一些硬件?谁能为业余爱好者推荐一些便宜的东西,以及一些更专业的东西? 我认为会有一个带有硬件的 API 的完整文档,所以也许这
我使用 python 制作了一个简单的二十一点游戏。我制作了游戏的其余部分,但我正在努力放入 ASCII 卡,所以这只是代码的一小部分。我尝试将 * len(phand) 放在附加行的末尾。虽然这确实
我记得前一段时间读到有 cpu 卡供系统添加额外的处理能力来进行大规模并行化。任何人都有这方面的经验和任何资源来研究项目的硬件和软件方面吗?这项技术是否不如传统集群?它更注重功率吗? 最佳答案 有两个
我检查外部存储是否已安装并且可用于读/写,然后从中读取。我使用的是确切的官方 Android 示例代码 ( from here )。 它说外部存储未安装。 getExternalFilesDir(nu
在 Android 2.1 及更低版本中,Android 应用程序可以请求下载到 SD 卡上吗?另外我想知道应用程序是否可以请求一些包含视频的文件夹下载到 SD 卡上?以及如何做到这一点? 提前致谢。
我们编写了一个 Windows 设备驱动程序来访问我们的自定义 PCI 卡。驱动程序使用 CreateFile 获取卡的句柄。 我们最近在一次安装中遇到了问题,卡似乎停止工作了。我们尝试更换卡(更换似
有些新设备(例如 Samsung Galaxy)带有两个 SD 卡。我想知道是否有任何方法可以确定设备是否有两张 SD 卡或一张 SD 卡。谢谢 最佳答案 我认为唯一的方法是使用 检查可用根的列表 F
我正在尝试将文件读/写到 SD 卡。我已经尝试在我的真实手机和 Eclipse 中的模拟器上执行此操作。在这两种设备上,对/mnt/sdcard/或/sdcard 的权限仅为“d--------”,我
我是一名优秀的程序员,十分优秀!