- 在另外一个应用程序里面 通过contentResolver 增删改查
7.内容提供者编写的流程
- 创建一个类继承ContentProvider
public class BankDBBackdoor extends ContentProvider {
}
- 在清单文件的application节点中进行配置
<application
android:allowBackup=“true”
android:icon=“@drawable/ic_launcher”
android:label=“@string/app_name”
android:theme=“@style/AppTheme” >
…
…
<provider
android:name=“com.mythmayor.db.BankDBBackdoor”
//必须配置该主机名,访问者使用该主机名才能访问
android:authorities=“com.mythmayor.db” >
- 在内容提供者代码的内部声明UriMatcher,创建匹配规则
public static final int SUCCESS = 1;
/**
- 创建一个保安,检查uri的规则,如果uri匹配失败 返回-1
*/
static UriMatcher mUriMatcher = new UriMatcher(UriMatcher.NO_MATCH);
static {
mUriMatcher.addURI(“com.mythmayor.db”, “account”, SUCCESS);
}
-
实现增删改查的方法,通过uriMatcher的返回值确定要做什么操作
-
在另外一个应用程序里面,通过contentResolver进行增删改查
8.学习内容提供者的目的
-
了解内容提供者原理
-
能看懂系统源码
-
获取系统应用内容提供者所提供的数据,例如联系人、短信应用
9.如何去分析系统应用的内容提供者
-
查看数据库,分析数据库的表和字段
-
操作内容提供者需要uri
-
找到系统应用的源代码,首先去清单文件中查找主机名authorities
<provider android:name=“SmsProvider”
android:authorities=“sms”
android:multiprocess=“true”
android:readPermission=“android.permission.READ_SMS”
android:writePermission=“android.permission.WRITE_SMS” />
- 去对应的Provider的源代码中查找匹配规则,确定表名
static {
sURLMatcher.addURI(“sms”, null, SMS_ALL); //所有短信
sURLMatcher.addURI(“sms”, “inbox”, SMS_INBOX); //收件箱
sURLMatcher.addURI(“sms”, “sent”, SMS_SENT); //发件箱
sURLMatcher.addURI(“sms”, “draft”, SMS_DRAFT); //草稿箱
}
- 根据主机名和表名确定uri,使用ContentResolver的增删改查方法操作对应的数据库
10.通知栏提醒Notification
显示在另外一个进程的界面里面的
- 在低版本中的写法(api小于16),创建Notification时直接new Notification()
NotificationManager nm = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
//1.初始化Notification
Notification notification = new Notification(R.drawable.ic_launcher, “有新的消息到来了”, System.currentTimeMillis());
//2.创建通知栏的点击事件
Intent intent = new Intent();
intent.setAction(Intent.ACTION_CALL);
intent.setData(Uri.parse(“tel://110”));
//PendingIntent延时的意图,可以打开Activity、Service和发送广播
PendingIntent contentIntent = PendingIntent.getActivity(this, 0, intent, 0);
//3.设置通知的点击事件
notification.setLatestEventInfo(this, “我是标题”, “我是文本”, contentIntent);
//4.显示通知
nm.notify(0, notification);
- 在高版本中的写法(api大于等于16),创建Notification时使用Notification.Builder
NotificationManager nm = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
//1.初始化Notification
Notification notification = new Notification.Builder(this)
.setContentTitle(“我是标题”)
.setContentText(“我是文本”)
.setSmallIcon(R.drawable.ic_launcher)
.setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.ic_launcher))
.setContentIntent(PendingIntent intent) //设置点击事件
.build();
//2.显示通知
nm.notify(0, notification);
- Notification中使用自定义View
api小于16
Notification notification = new Notification(R.drawable.ic_launcher, “有新的消息到来了”, System.currentTimeMillis());
//设置自定义布局
RemoteViews remoteViews = new RemoteViews(getPackageName(), R.layout.xxx);
notification.contentView = remoteViews;
api大于16
Builder builder = Notification.Builder(this);
//设置自定义布局
builder.setContent(RemoteViews views);
- RemoteViews使用方式
RemoteViews views = new RemoteViews(getPackageName(), R.layout.xxx);
views.setTextViewText(R.id.xxx, “hello”); //设置TextView的文本
views.setImageViewResource(R.id.xxx, R.drawable.xxx); //设置ImageView的图片
views.setOnClickPendingIntent(R.id.xx, pendingIntent); //设置按钮的点击事件
11.如何打开短信界面
尝试打开系统某个界面的思路
-
从logcat中查看是哪个Activity
-
在上层源码中搜索该工程
-
查看清单文件中是否有隐式意图可以激活
//打开短信界面的隐式意图
Intent intent = new Intent();
intent.setAction(“android.intent.action.MAIN”);
intent.addCategory(“android.intent.category.DEFAULT”);
intent.setType(“vnd.android.cursor.dir/mms”);
startActivity(intent);
12.联系人数据库
-
路径:data/data/com.android.providers.contacts/databases/contacts2.db
-
主要操作的3张表:
-
raw_contact:联系人的id表
- contact_id 保存联系人的id
- data:联系人的数据表
-
raw_contact_id 表示属于哪个联系人
-
data1 具体的数据
-
mimetype_id 数据的类型,使用该id去mimetypes表中查询数据类型
-
mimetypes:联系人的数据类型表
-
查询联系人数据库的数据的步骤
-
查询raw_contact表,获取所有联系人id
-
根据联系人id,查询data表,该联系人的所有数据
-
根据mimetype确定数据类型
13.如何读取联系人数据
ContentResolver resolver = getContentResolver();
//1.查询raw_contact表,获取所有联系人id
Uri uri = Uri.parse(“content://com.android.contacts/raw_contacts”);
Uri datauri = Uri.parse(“content://com.android.contacts/data”);
Cursor cursor = resolver.query(uri, new String[]{“contact_id”}, null, null, null);
while(cursor.moveToNext()){
String id = cursor.getString(0);
System.out.println(“Id:”+id);
//2.根据联系人id,查询data表,该联系人的所有数据
Cursor datacursor = resolver.query(datauri, new String[]{“data1”,“mimetype”}, “raw_contact_id=?”, new String[]{id}, null);
while(datacursor.moveToNext()){
//3.根据mimetype确定数据类型
String data1 = datacursor.getString(0);
System.out.println(“data1:”+data1);
String mimetype = datacursor.getString(1);
System.out.println(“mimetype:”+mimetype);
}
datacursor.close();
System.out.println(“------------”);
}
cursor.close();
-
备注:加上READ_CONTACTS权限,取mimetype时有个小细节,直接在data表里面就可以查询mimetype数据类型了,实际是查询数据库中的视图。
-
数据库视图:视图是虚表,是从一个或几个基本表(或视图)中导出的表。
-
简单性:看到的就是需要的。视图不仅可以简化用户对数据的理解,也可以简化他们的操作。那些被经常使用的查询可以被定义为视图。
-
安全性:通过视图用户只能查询和修改他们所能见到的数据。数据库中的其它数据则既看不见也取不到。
14.系统联系人应用删除一个联系人的处理逻辑
-
联系人应用删除一个联系人时,只是在raw_contact表中,把对应联系人的id置为null了
-
Google这么设计的目的是为了进行联系人与服务器同步,还有减少计算量
-
比如,本地有4个联系人,服务器有5个联系人,而且服务器也具有添加联系人的功能,那同步时的逻辑是要往本地加一个联系人呢?还是要在服务器减一个联系人呢?
-
另外对比少哪一个数据是一个非常麻烦的算法,要从A里取一个去遍历B里是否有这样一个数据
15.添加一个联系人到数据库的步骤
-
在raw_contact表中添加一个联系人的id
-
在data表里添加联系人的数据,姓名、电话、邮箱等
16.内容观察者
//注册内容观察者
Uri uri = Uri.parse(“content://com.mythmayor.db/account”);
getContentResolver().registerContentObserver(uri, true, new ContentObserver(new Handler()) {
@Override
public void onChange(boolean selfChange) {
System.out.println(“我是观察者,我发现银行的数据库变化了.”);
super.onChange(selfChange);
}
});
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
最后
针对Android程序员,我这边给大家整理了一些资料,包括不限于高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter等全方面的Android进阶实践技术;希望能帮助到大家,也节省大家在网上搜索资料的时间来学习,也可以分享动态给身边好友一起学习!
往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、混合式开发(ReactNative+Weex)全方面的Android进阶实践技术,群内还有技术大牛一起讨论交流解决问题。
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
g2.imgtp.com/2024/03/13/H4lCoPEF.jpg" />
最后
针对Android程序员,我这边给大家整理了一些资料,包括不限于高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter等全方面的Android进阶实践技术;希望能帮助到大家,也节省大家在网上搜索资料的时间来学习,也可以分享动态给身边好友一起学习!
往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、混合式开发(ReactNative+Weex)全方面的Android进阶实践技术,群内还有技术大牛一起讨论交流解决问题。
[外链图片转存中…(img-jSlTpe71-1712605849501)]
[外链图片转存中…(img-hIVJnllU-1712605849501)]
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!