`

android短彩信查询以及MMS表结构

 
阅读更多
android短信的数据库的Uri是不公开的, 读取起来时灰常不方便的, 这里做了下总结. 
用adb指令将mmssms.db从/data/data/com.android.providers.telephony/databases中pull出来
经常使用到的表有
canonical_addresses, sms, threads三个表格
sms是存储着所有的短信, 主要的列有_id, thread_id, address, person, date, read, type, body
关于的sms的Uri有
发件箱 content://sms/outbox
收件箱 content://sms/inbox
草稿箱 content://sms/draft
conversations content://sms/conversations
threads表存储着每一个短信对话的线程. 主要列有_id, date, message_count, recipient_ids, snippet, read
recipient_ids 存放的是参与此次对话的person的id, 然而这个id不是通讯录里面的id, 而是canonical_addresses 的id. 这就是canonical_addresses 表格的作用
threads 表 uri: content://mms-sms/conversations?simple=true
canonical_addresses 表 uri content://mms-sms/canonical-addresses

一、需要实现一个同系统信息一样的功能

1)会话列表;

2)会话对话详情;

3)系统会话增加新信息或者删除信息等变化时做到同步;

二、实现思路

通过查询会话表显示会话界面,监听会话数据库实现与系统信息同步。通过会话id查询会话对应的具体聊天内容,比如短信或者彩信;

看下mmssms.db中的uri有好多个,可以根据自己的需求选择合适的uri,可以避免不必要的操作,提高效率。经过多次测试,实现以上功能需要用的信息表有:

1)会话表:content://mms-sms/conversations/

2)短信表:content://sms

3)彩信表:content://mms、content://mms/part

4)信息对应手机号码表:content://mms-sms/canonical-addresses

需要权限:<uses-permission android:name="android.permission.READ_SMS" />





首先:显示会话列表

public static final Uri THREADS_URI = Uri.parse("content://mms-sms/conversations?simple=true");


threads字段:

_id
date
ct_t:区分彩信还是短信:application/vnd.wap.multipart.related是彩信,否则是短信
snippet:最新的一条会话信息。彩信为彩信的主题,短信时短信的body
recipient_ids:信息对应手机号码表(content://mms-sms/canonical-addresses)中的id,查询对应的手机号码


message_count:详细对话条数


read


snippet_cs:snippet的编码方式,彩信:106(utf-8),短信:0

……

根据threads表中的date,snippet,messagecount基本可以显示会话列表了。




另外:

1:通过4)信息对应手机号码表:content://mms-sms/canonical-addresses获取了手机号之后获取联系人姓名以及头像使用:ContactsContract.CommonDataKinds.Phone.CONTENT_URI

2:当会话为彩信时,snippet显示的是彩信的主题,编码方式为utf-8。显示为乱码。

解决:snippet_mms = new String(snippet.getBytes("ISO8859_1"), "utf-8");




其次,获取会话具体对话内容

具体对话的格式有ssm或者mms,通过content://mms-sms/conversations/threadid,可以查询与某联系人的所有聊天记录的id,这个id对应短信或者彩信表中的id。如果具体会话是短信,则去sms表中查询具体内容,如果是彩信则去mms表中查询。通过字段ct_t 区分信息类别。

具体实现:

1)获取sms


[java] view plaincopyprint?String selection = "_id = "+id; 
Uri uri = Uri.parse("content://sms"); 
Cursor cursor = contentResolver.query(uri, null, selection, null, null); 
String phone = cursor.getString(cursor.getColumnIndex("address")); 
int type = cursor.getInt(cursor.getColumnIndex("type"));// 2 = sent, etc.  
String date = cursor.getString(cursor.getColumnIndex("date")); 
String body = cursor.getString(cursor.getColumnIndex("body")); 

String selection = "_id = "+id;
Uri uri = Uri.parse("content://sms");
Cursor cursor = contentResolver.query(uri, null, selection, null, null);
String phone = cursor.getString(cursor.getColumnIndex("address"));
int type = cursor.getInt(cursor.getColumnIndex("type"));// 2 = sent, etc.
String date = cursor.getString(cursor.getColumnIndex("date"));
String body = cursor.getString(cursor.getColumnIndex("body"));
2)获取mms

彩信表:content://mms

彩信附件:content://mms/part/


get text content from MMS:
[java] view plaincopyprint?String selectionPart = "mid=" + mmsId; 
Uri uri = Uri.parse("content://mms/part"); 
Cursor cursor = getContentResolver().query(uri, null, 
    selectionPart, null, null); 
if (cursor.moveToFirst()) { 
    do { 
        String partId = cursor.getString(cursor.getColumnIndex("_id")); 
        String type = cursor.getString(cursor.getColumnIndex("ct")); 
        if ("text/plain".equals(type)) { 
            String data = cursor.getString(cPart.getColumnIndex("_data")); 
            String body; 
            if (data != null) { 
                // implementation of this method below  
                body = getMmsText(partId); 
            } else { 
                body = cursor.getString(cursor.getColumnIndex("text")); 
            } 
        } 
    } while (cursor.moveToNext()); 


String selectionPart = "mid=" + mmsId;
Uri uri = Uri.parse("content://mms/part");
Cursor cursor = getContentResolver().query(uri, null,
    selectionPart, null, null);
if (cursor.moveToFirst()) {
    do {
        String partId = cursor.getString(cursor.getColumnIndex("_id"));
        String type = cursor.getString(cursor.getColumnIndex("ct"));
        if ("text/plain".equals(type)) {
            String data = cursor.getString(cPart.getColumnIndex("_data"));
            String body;
            if (data != null) {
                // implementation of this method below
                body = getMmsText(partId);
            } else {
                body = cursor.getString(cursor.getColumnIndex("text"));
            }
        }
    } while (cursor.moveToNext());
}


[java] view plaincopyprint?private String getMmsText(String id) { 
    Uri partURI = Uri.parse("content://mms/part/" + id); 
    InputStream is = null; 
    StringBuilder sb = new StringBuilder(); 
    try { 
        is = getContentResolver().openInputStream(partURI); 
        if (is != null) { 
            InputStreamReader isr = new InputStreamReader(is, "UTF-8"); 
            BufferedReader reader = new BufferedReader(isr); 
            String temp = reader.readLine(); 
            while (temp != null) { 
                sb.append(temp); 
                temp = reader.readLine(); 
            } 
        } 
    } catch (IOException e) {} 
    finally { 
        if (is != null) { 
            try { 
                is.close(); 
            } catch (IOException e) {} 
        } 
    } 
    return sb.toString(); 


private String getMmsText(String id) {
    Uri partURI = Uri.parse("content://mms/part/" + id);
    InputStream is = null;
    StringBuilder sb = new StringBuilder();
    try {
        is = getContentResolver().openInputStream(partURI);
        if (is != null) {
            InputStreamReader isr = new InputStreamReader(is, "UTF-8");
            BufferedReader reader = new BufferedReader(isr);
            String temp = reader.readLine();
            while (temp != null) {
                sb.append(temp);
                temp = reader.readLine();
            }
        }
    } catch (IOException e) {}
    finally {
        if (is != null) {
            try {
                is.close();
            } catch (IOException e) {}
        }
    }
    return sb.toString();
}

获取彩信中图片:


[java] view plaincopyprint?String selectionPart = "mid=" + mmsId; 
Uri uri = Uri.parse("content://mms/part"); 
Cursor cPart = getContentResolver().query(uri, null, 
    selectionPart, null, null); 
if (cPart.moveToFirst()) { 
    do { 
        String partId = cPart.getString(cPart.getColumnIndex("_id")); 
        String type = cPart.getString(cPart.getColumnIndex("ct")); 
        if ("image/jpeg".equals(type) || "image/bmp".equals(type) || 
                "image/gif".equals(type) || "image/jpg".equals(type) || 
                "image/png".equals(type)) { 
            Bitmap bitmap = getMmsImage(partId); 
        } 
    } while (cPart.moveToNext()); 


String selectionPart = "mid=" + mmsId;
Uri uri = Uri.parse("content://mms/part");
Cursor cPart = getContentResolver().query(uri, null,
    selectionPart, null, null);
if (cPart.moveToFirst()) {
    do {
        String partId = cPart.getString(cPart.getColumnIndex("_id"));
        String type = cPart.getString(cPart.getColumnIndex("ct"));
        if ("image/jpeg".equals(type) || "image/bmp".equals(type) ||
                "image/gif".equals(type) || "image/jpg".equals(type) ||
                "image/png".equals(type)) {
            Bitmap bitmap = getMmsImage(partId);
        }
    } while (cPart.moveToNext());
}
[java] view plaincopyprint?private Bitmap getMmsImage(String _id) { 
    Uri partURI = Uri.parse("content://mms/part/" + _id); 
    InputStream is = null; 
    Bitmap bitmap = null; 
    try { 
        is = getContentResolver().openInputStream(partURI); 
        bitmap = BitmapFactory.decodeStream(is); 
    } catch (IOException e) {} 
    finally { 
        if (is != null) { 
            try { 
                is.close(); 
            } catch (IOException e) {} 
        } 
    } 
    return bitmap; 


private Bitmap getMmsImage(String _id) {
    Uri partURI = Uri.parse("content://mms/part/" + _id);
    InputStream is = null;
    Bitmap bitmap = null;
    try {
        is = getContentResolver().openInputStream(partURI);
        bitmap = BitmapFactory.decodeStream(is);
    } catch (IOException e) {}
    finally {
        if (is != null) {
            try {
                is.close();
            } catch (IOException e) {}
        }
    }
    return bitmap;
}
测试结果:








最后,我的会话界面与系统的实时同步

监听系统的会话数据表即可:

public static final Uri THREADS_URI = Uri
.parse("content://mms-sms/conversations?simple=true");


getContentResolver().registerContentObserver(THREADS_URI ,true, sysCallLogObserver);




另外:

我只写了自己用的表字段,如果想字段别的可以自己打印下:

cursor.getColumnNames();
========================================
数据库中sms相关的字段如下:   



_id                         primary key     integer                  与words表内的source_id关联

thread_id              会话id,一个联系人的会话一个id,与threads表内的_id关联      integer

address                 对方号码          text

person                  联系人id           integer            

date                      发件日期           integer

protocol               通信协议,判断是短信还是彩信     integer       0:SMS_RPOTO, 1:MMS_PROTO

read                      是否阅读           integer       default 0             0:未读, 1:已读 

status                   状态                  integer       default-1           -1:接收,

                                                                                                   0:complete,

                                                                                                 64: pending,

                                                                                                128: failed

type                     短信类型           integer                                   1:inbox

                                                                                                    2:sent

                                                                                                    3:draft56

                                                                                                    4:outbox

                                                                                                    5:failed

                                                                                                    6:queued           

body                      内容

service_center      服务中心号码

subject                  主题

reply_path_present

locked

error_code

seen
分享到:
评论

相关推荐

    计算机视觉开发:OpenCV入门教程及应用

    内容概要:本文档详细介绍了OpenCV的基本概念及其在计算机视觉领域的应用,重点讲解了OpenCV在C++和Python环境下的安装方法,并提供了图像读取、显示、基本操作、视频处理以及面部检测的具体代码示例。此外,还涉及了一些图像处理技术的快速演示和进一步学习的路径建议。 适合人群:对计算机视觉感兴趣的新手开发者和技术爱好者。 使用场景及目标:本教程适用于希望入门计算机视觉和图像处理的新手,通过实际操作练习提升技术水平,掌握OpenCV的基本用法,并能够应用于实际项目,如OCR应用、图像分割与目标检测等。 阅读建议:建议读者按照文档提供的步骤进行实践,逐步完成每个代码示例,结合官方文档和其他资源深入理解各个函数的作用。对于初学者来说,可以通过多动手尝试,加深对OpenCV的理解。

    围绕着一系列的经典Python练习题 .zip

    围绕着一系列的经典Python练习题。Python练习一些按照回顾排列的Python练习题。欢迎提交你的答案或添加更多有趣的题目!从开始学Python以来,接触了精彩的练习题。下面十个练习题,是我做的和练习出来的题里比较有趣的,现在按照难度由低到高排列。欢迎到GitHub上提交你的答案。猜测数字经典的猜数字游戏,几乎是学编程时都会做的。功能描述随机选择三个以内的数字作为答案。用户输入一个数字,程序会提示大了或者小了,直到用户猜中。2.FizzBu​​zz另一道经典编程题。功能描述遍历并打印0到100,如果数字能被3整除,显示Fizz如果数字能被5整除,显示Buzz如果能同时被3和5整除,就显示FizzBu​​zz。结果应该类似0,1 ,2,嘶嘶声,4,嗡嗡声,6……14,嘶嘶声,16……3. 猜测数字的AI和猜数字一样,不过这次是设计一个能猜数字的人工智能功能描述用户输入一个单位以内的数字,AI需要最少的猜测次数,并显示出猜测的次数和数字。4.整点报时老式的挂钟会在整点报时,响铃的次数和时间是一致的。我们设计了一个在电脑上运行的报时

    毕设源码-python-django基于python技术的学生管理系统的设计与开发-期末大作业+说明文档.rar

    本项目是一个基于Python技术的学生管理系统,采用Django框架进行开发,旨在为计算机相关专业的学生提供一个实践性强、功能全面的管理系统,以帮助他们完成毕业设计或进行项目实战练习。 系统实现了对学生信息、课程信息、成绩、考勤等多方面的管理功能。学生信息管理包括学生基本信息的增删改查;课程信息管理允许管理员设置课程信息,包括课程名称、授课老师、学分等;成绩管理功能使学生和教师能够录入、查看和修改成绩;考勤管理则方便教师记录学生的出勤情况。 该项目采用B/S架构,前端使用HTML、CSS、JavaScript等技术,后端使用Python语言和Django框架,数据库采用MySQL。Django框架提供了强大的后台管理功能,使得系统管理更加便捷。 通过开发这个项目,学生不仅能提升自己的编程能力,还能学习到如何构建一个实际应用的系统,对于即将步入职场的学生来说,具有很高的实用价值。

    python入门-安装Python软件包.pdf

    python入门——安装Python软件包

    消息中间件源码学习(打注释学习).zip

    消息中间件源码学习(打注释学习)

    阿里消息中间件MetaQ学习Demo.zip

    阿里消息中间件MetaQ学习Demo

    数学建模培训资料 数学建模实战题目真题答案解析解题过程&论文报告 抑制房地产泡沫问题研究 共69页.pdf

    数学建模培训资料 数学建模实战题目真题答案解析解题过程&论文报告 抑制房地产泡沫问题研究 共69页.pdf

    rbac组件(基于角色的权限控制).zip

    rbac组件(基于角色的权限控制)

    Discuz! X3.5 R20240520 增量补丁包下载 X3.5 2023-12-21 升级到 X3.5 2024-05-20 补丁包

    压缩包内,SC是简体补丁包,TC是繁体补丁包

    ssm框架Java项目源码-基于Java的在线日语培训平台的设计与实现+jsp毕设-大作业.zip

    本项目是一个基于Java的在线日语培训平台的设计与实现,采用SSM框架(Spring+SpringMVC+MyBatis)进行开发,旨在为计算机相关专业的学生提供一个实践和学习的平台,同时也为日语学习者提供一个在线学习的空间。项目中主要功能涵盖了用户管理、课程管理、学习资源上传下载、在线测试与反馈等多个方面。通过该平台,教师能够轻松管理课程内容和学生信息,学生则可以随时随地访问学习资源,参与在线课程和测试,从而提高学习效率和兴趣。 在开发此项目的过程中,我们重点关注了系统的可维护性和可扩展性,确保代码结构清晰,便于后续的功能迭代和优化。此外,通过使用SSM框架,实现了前后端的分离,提高了开发效率和系统的响应速度。该项目不仅能够满足毕设的需求,还能作为Java学习者提升编程能力和实践经验的实用工具。

    机器xuex(大模型):语言模型在生成问题答案时的真实性数据集

    TruthfulQA是一个专门设计的基准测试数据集,用于衡量。这个数据集包含了817个问题,覆盖了38个不同的类别,如健康、法律、金融和政治等。这些问题被精心设计,以至于某些人可能会因为错误的信念或误解而给出错误的答案。因此,要在这个数据集上表现良好,语言模型必须避免生成从模仿人类文本中学到的错误答案。 TruthfulQA的数据集结构包括两种配置:generation和multiple_choice。在generation配置中,每个问题都包含了类型、类别、问题、最佳答案、正确答案列表、错误答案列表和来源。而在multiple_choice配置中,每个问题都提供了四个选项,模型需要从中选择正确的答案。 这个数据集的目的是为了测试语言模型在真实性方面的弱点,而不是测试模型在有用任务上的表现。研究发现,最大的模型通常是最不真实的,这与其他NLP任务不同,在其他任务中,模型的性能随着模型大小的增加而提高。TruthfulQA的数据集提供了一个重要的工具,用于评估和改进语言模型在生成真实和可靠信息方面的能力。

    多彩吉安红色旅游网站-JAVA-基于springBoot多彩吉安红色旅游网站的设计与实现

    多彩吉安红色旅游网站-JAVA-基于springBoot多彩吉安红色旅游网站的设计与实现

    基于servlet+jsp+mysql实现的影视管理系统课程设计

    【作品名称】:基于servlet+jsp+mysql实现的影视管理系统【课程设计】 【适用人群】:适用于希望学习不同技术领域的小白或进阶学习者。可作为毕设项目、课程设计、大作业、工程实训或初期项目立项。 【项目介绍】: 基于servlet+jsp+mysql实现的影视管理系统【课程设计】 基于servlet+jsp+mysql实现的影视管理系统【课程设计】 Java Web课程设计,基于servlet+jsp+ajax+mysql做的影视管理系统 运行环境: Tomcat 9.0 JDK 1.8 MySQL 8.0 后台管理账号密码均为:root,项目依赖:lib 目录 【资源声明】:本资源作为“参考资料”而不是“定制需求”,代码只能作为参考,不能完全复制照搬。需要有一定的基础看懂代码,自行调试代码并解决报错,能自行添加功能修改代码。

    渗透测试人员的 Python 工具.zip

    渗透测试人员的 Python 工具渗透测试人员的 Python 工具如果你参与漏洞研究、逆向工程或渗透测试,我建议尝试 Python编程语言。它有一套丰富的有用库和程序。本页列出了其中一些。列出的大多数工具都是用 Python 编写的,其他工具只是现有 C 库的 Python 绑定,即它们使这些库可轻松地在 Python 程序中使用。一些更激进的工具(渗透测试框架、蓝牙粉碎器、Web 应用程序漏洞扫描器、战争拨号器等)被排除在外,因为这些工具在德国的法律地位仍然不太明确——即使在最高法院作出裁决之后也是如此。这个列表显然是为了帮助白帽黑客,目前我更愿意谨慎行事。网络Scapy发送、嗅探、剖析和伪造网络数据包。可交互使用或作为库使用pypcap、 Pcapy和 pylibpcaplibpcap 的几种不同的 Python 绑定libdnet低级网络例程,包括接口查找和以太网帧传输dpkt快速、简单的数据包创建/解析,包含基本 TCP/IP 协议的定义Impacket制作和解码网络数据包。包括对 NMB 和 SMB 等高级协议的支持pynidslibnids

    quark(夸克)正版下载

    quark(夸克)正版下载

    ssm框架Java项目源码-企业员工岗前培训管理系统+vue毕设-大作业.zip

    ssm框架Java项目源码-企业员工岗前培训管理系统+vue毕设-大作业.zip是一个专为计算机相关专业学生和Java学习者设计的项目资源。该项目以企业员工岗前培训管理为背景,采用经典的SSM(Spring+SpringMVC+MyBatis)框架进行后端开发,确保系统的稳定性和可扩展性。同时,前端采用Vue.js框架,实现了前后端分离,提升了用户体验和开发效率。 该项目的主要功能包括员工信息管理、培训课程管理、培训进度跟踪、考试成绩记录与统计等。通过这些功能,系统能够帮助企业高效地管理员工的岗前培训过程,确保培训质量,提升员工技能水平。 此外,该项目不仅适合作为计算机专业学生的毕业设计题目,也适合Java学习者进行项目实战练习,通过实际操作,加深对SSM框架和Vue.js的理解,提升编程能力和解决问题的能力。

    汇聚【Python应用】【Python实训】【Python技术分享】等等.zip

    你是 Pythonista汇聚【从零单排】【实战项目】【数据科学】【自然语言处理】【计算机】【面试题系列】【大航海】【Python应用】【错题集】【技术沙龙】【内推渠道】 】等等【人人都是Pythonista】由公众号【Python专栏】推出,请认准唯一标识请仔细阅读本文档,尤其是使用说明。目录说明计算机视觉计算机视觉DataScience数据科学作业所有的作业都提交在这个目录下,每个人创建属于自己的独立目录HR内推渠道Interview_Questions: 面试题集KnowledgeShare干货分享LearnFromZero从零单排NLP自然语言处理、自然语言处理OnePiece大航海PracticeProject实战项目PythonExercisePython练习、应用资源资源目录TechSalon技术沙龙WeeklyReport每周新鲜事、分享好项目、好资源使用说明命名规范文件夹命名规范必须为英文全部小写单词之间用下划线分割,例如nagios_check_tomcat第一个单词代表项目涉及应用,若有多个采用简写,最多2

    基于java的学生社团管理系统设计与实现.docx

    基于java的学生社团管理系统设计与实现.docx

    学习微服务框架SpringCloud的一些示例代码.zip

    学习微服务框架SpringCloud的一些示例代码

    基于java的数字家庭网站设计与实现.docx

    基于java的数字家庭网站设计与实现.docx

Global site tag (gtag.js) - Google Analytics