Android内部分享[4]——Intent 和 Intent 过滤器

概述

前面我们已经对 Android 中的界面跳转和本地存储有了一定的认识,接下来我们来详细研究一下 Android 中的意图(Intent),理解它有助于我们后续理解系统广播和服务。

我们先来回顾一下前面的界面跳转代码:

1
2
Intent intent = new Intent(MainActivity.this, SecondActivity.class);
startActivity(intent);

可以看到 Intent 就像一座桥梁,来实现从一个界面到另一个界面的切换,事实上上面的是显式的 Intent,我们很明确我们即将要跳转到的页面,还有一种意图是不可知不确定的,或者说在绑定意图的时候不是固定的某个具体的 Activity, 例如:

1
2
3
4
5
Intent intent = new Intent();
sendIntent.setAction(Intent.ACTION_SEND);
sendIntent.putExtra(Intent.EXTRA_TEXT, textMessage);
sendIntent.setType("text/plain");
startActivity(intent);

这个时候我们的 Intent (意图)可能指定的不是一个界面了,有可能是多个界面,只要我们的 Activity 含 ACTION_SEND 操作并携带 text/plain 数据就会被调起。如果只有一个应用能够处理,则该应用将立即打开并为其提供 Intent。 如果多个 Activity 接受 Intent,则系统将显示一个对话框,使用户能够选取要使用的应用。

多个意图响应

携带的数据

Intent 对象携带了 Android 系统用来确定要启动哪个组件的信息(例如,准确的组件名称或应当接收该 Intent 的组件类别),以及收件人组件为了正确执行操作而使用的信息(例如,要采取的操作以及要处理的数据)。

组件名称:要启动的组件名称(这里的组件不仅仅指 Activity, 还包括 Service)。

这虽然在 Intent 构造函数中是一个可选项,但是在显式意图中是一个很重要的信息,如果没有这个组件名称就是隐式的意图。

1
2
3
Intent intent = new Intent();
intent.setClassName(this, SecondActivity.class.getName());
startActivity(intent);

也可以使用上面这种 setClassName 来设置组件名称, 第一个参数可以是 context 也可以是 packageName, 第二个参数是组件的全路径字符串, 和下方代码是等价的:

1
intent.setClassName(this, "com.dlc.helloword.activity.SecondActivity");

或者:

1
intent.setClassName(getPackageName(), "com.dlc.helloword.activity.SecondActivity");

实际上在 Intent 中表示组件名称的对象是 ComponentName, 上面的所有写法只不过是提供给我们的简写方式,使用该对象设置组件名称如下:

ComponentName 构造函数

1
2
3
4
5
Intent intent = new Intent();
ComponentName componentName = new ComponentName(getPackageName(),
"com.dlc.helloword.activity.SecondActivity");
intent.setComponent(componentName);
startActivity(intent);

同理,根据构造函数我们可以传递 context 和 Class 对象。

ACTION : 指定要执行的通用操作(例如,查看选取)的字符串。

你可以在 AndroidManifest.xml 文件中定义自己的 ACTION, 就像我们的入口 Activity 一样。

1
2
3
4
5
6
7
<activity android:name=".activity.MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />

<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>

上面入口函数定义的 ACTION 是 android.intent.action.MAIN, 但是大多数情况下我们会使用系统定义的 ACTION 常量, 在 Intent 类中定义了很多 ACTION 常量如下:

Intent 中的 ACTION 常量

例如 ACTION_VIEW 向用户显示信息,例如照片, ACTION_SEND 共享数据等,在 Intent 中对 ACTION 进行了分类:

标准的 Activity ACTIONS:

  • ACTION_MAIN:入口Activity 不期望接收数据。
  • ACTION_VIEW:最常见的操作,向用户显示数据。
  • ACTION_ATTACH_DATA
  • ACTION_EDIT
  • ACTION_PICK
  • ACTION_CHOOSER
  • ACTION_GET_CONTENT
  • ACTION_DIAL
  • ACTION_CALL
  • ACTION_SEND
  • ACTION_SENDTO
  • ACTION_ANSWER
  • ACTION_INSERT
  • ACTION_DELETE
  • ACTION_RUN
  • ACTION_SYNC
  • ACTION_PICK_ACTIVITY
  • ACTION_SEARCH
  • ACTION_WEB_SEARCH
  • ACTION_FACTORY_TEST

标准的 Broadcast(广播)ACTIONS:

  • ACTION_TIME_TICK
  • ACTION_TIME_CHANGED
  • ACTION_TIMEZONE_CHANGED
  • ACTION_BOOT_COMPLETED
  • ACTION_PACKAGE_ADDED
  • ACTION_PACKAGE_CHANGED
  • ACTION_PACKAGE_REMOVED
  • ACTION_PACKAGE_RESTARTED
  • ACTION_PACKAGE_DATA_CLEARED
  • ACTION_PACKAGES_SUSPENDED
  • ACTION_PACKAGES_UNSUSPENDED
  • ACTION_UID_REMOVED
  • ACTION_BATTERY_CHANGED
  • ACTION_POWER_CONNECTED
  • ACTION_POWER_DISCONNECTED
  • ACTION_SHUTDOWN

例如我们打开浏览器访问网站,或者打开键盘拨号界面:

1
2
3
4
5
6
7
public void testIntent(View view){
Uri uri = Uri.parse("https://dp2px.com"); //浏览器
//Uri uri = Uri.parse("tel:1232333"); //拨号程序
//Uri uri= Uri.parse("geo:39.899533,116.036476"); //打开地图定位
Intent intent = new Intent(Intent.ACTION_VIEW, uri); //Intent.ACTION_VIEW不带引号
startActivity(intent);
}

Data:引用待操作数据和/或该数据 MIME 类型的 URI(Uri 对象)。提供的数据类型通常由 Intent 的操作决定。

1
2
3
4
Intent intent = new Intent();
intent.setAction(Intent.ACTION_VIEW);
intent.setData(Uri.parse("https://dp2px.com"));
startActivity(intent);

如果要设置 MIME 类型,调用 setType() 方法。

Category:一个包含应处理 Intent 组件类型的附加信息的字符串。

还记得上面的入口 Activity 注册表中的 category 吗?

1
2
3
4
5
6
<activity android:name=".activity.MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>

这个 category 表示此 Activity 应该显示在 launcher 的顶层。当然我们也可以自己去设定 category 去分门别类,这个就有点像给 Activity 添加标签。通过 addCategory() 方法来启动该标签的 Activity。

过滤器

上面的 action, category, data 都需要在 <intent-filter> 中添加,而且可以添加多个,例如:

1
2
3
4
5
6
7
8
<intent-filter>
<action android:name="android.intent.action.SEND"/>
<action android:name="android.intent.action.SEND_MULTIPLE"/>
<category android:name="android.intent.category.DEFAULT"/>
<data android:mimeType="application/vnd.google.panorama360+jpg"/>
<data android:mimeType="image/*"/>
<data android:mimeType="video/*"/>
</intent-filter>

这样我们就可以从各个层面去过滤, 当我们调用 Intent 意图去跳转的时候,会坚持是不是 SEND 和 SEND_MULTIPLE 的 action, 数据是不是 image 或者 video 类型,如果不是则不会调起该 Activity.

PendingIntent

我们上面的 Intent 使用 startActivity() 方法可以理解执行意图,而有时候我们不需要立即执行,而是要等待一个条件去触发执行,这个时候就需要使用 PendingIntent.

PendingIntent 有三个静态方法,分别获得对应组件的 Intent 对象:

  • getActivity(Context, int, Intent, int):跳转到一个activity组件、
  • getBroadcast(Context, int, Intent, int):打开一个广播组件
  • getService(Context, int, Intent, int):打开一个服务组件。
1
2
3
4
5
6
7
8
9
10
11
12
private void showNotify(){   
Notification notice=new Notification();
notice.icon=R.drawable.icon;
notice.tickerText="您有一条新的信息";
notice.defaults=Notification.DEFAULT_SOUND;
notice.when=10L;
//即将跳转页面,还没跳转
notice.setLatestEventInfo(this, "通知", "开会啦",
PendingIntent.getActivity(this, 0, new Intent(this,Activity2.class), 0));
NotificationManager manager=(NotificationManager)getSystemService(this.NOTIFICATION_SERVICE);
manager.notify(0,notice);
}

例如上面是通知 Notification 点击跳转的一个例子,通常还常常用于短消息 SmsManager 的发送和警报器 AlarmManager 的执行等。

评论

3D Adapter Ajax Android AndroidStudio Animation Anroid Studio AppBarLayout AsyncTask Babel Banner Buffer Bulma ByteBuffer C++ C11 C89 C99 CDN CMYK CSS Camera Raw Canvas Chrome Class ContentProvider CoordinatorLayout C语言 DML DOM Dagger Dagger2 Darktable Demo Document DownloadManage ES2015 ESLint Element Elements Error Exception Extensions File FileProvider FileSave Flow Fresco GCC Git Git flow GitHub GitLab Github flow Gitlab flow Glide Gradle GrideView Groovy HTML HTML5 Handler HandlerThread Hexo Hilo Hybrid I/O IDEA IO ImageMagick ImageView IntelliJ Intellij Intent Interpolator JCenter JNI JS Java JavaScript JsBridge Kotlin Lab Lambda Lifecycle Lint Linux ListView Looper MQTT MVC MVP Maven MessageQueue Modbus Momentum MySQL NDK NIO Next Nodejs ObjectAnimator Okhttp Oracle VM Permission PhotoShop Physics Process Python RGB RS-232 RTU RecyclerView Remote-SSH Retrofit Runnable RxAndroid RxJava SE0 SQLite SSH SharePreference Spring SpringBoot Statubar Style Task Theme Thread Tkinter UI UIKit UML VM virtualBox VS Code VUE ValueAnimator ViewPropertyAnimator Vue Vue CLI Vue.js Web WebGL Web前端 Workbench Zdog api apk axios background blur bookmark by关键字 cli compileOnly component computed css css3 c语言 databases demo flex flexbox flow gradient hexo hotfix html iOS icarus icarus主题 implementation init jQuery javascript launchModel linear logo merge methods mvp offset photos pug query radial rxjava2 scss servlet shell slot slot-scope svg tkinter tomcat transition unicode utf-8 v-slot vector virtual box vscode watch webpack 七牛 下载 中介者模式 串口 临潼石榴 主题 书签 书籍 事件 享元模式 亮度 仓库 代理模式 位运算 依赖注入 修改,tables 光和色 入坑 入门视频教程 全套视频教程 内存 内核 内部分享 函数 函数式编程 分支 分析 列表 创建 删除 动画 单例模式 发布 可空性 合并 同向性 后期 启动模式 命令 命令模式 响应式 响应式编程 图层 图床 图形图像 图片压缩 图片处理 图片轮播 地球 域名 基础 增加 备忘录模式 外观模式 多线程 大爆炸 天气APP 太白山 头文件 奇点 字符串 字符集 存储引擎 宇宙 宏定义 实践 对比度 属性 属性动画 岐山擀面皮 岐山肉臊子 岐山香醋 工作流 工具 工厂模式 年终总结 开发技巧 开发流程 异常 弱引用 思想 恒星 打包 扫盲 技巧 指令 指针 接口 插件 插值 插槽 摄影 操作系统 攻略 故事 数据存储 数据库 数据类型 数组 文件 新功能 旅行 旋转木马 时序图 时空 时间简史 曲线 最佳实践 服务 杂谈 权限 枚举 架构 查询 标准库 标签选择器 样式 核心 框架 案例 桥接模式 检测工具 模块化 模板 模板引擎 模板方法模式 油泼辣子 泛型 洛川苹果 浅色状态栏 混合模式 渐变 渲染 游戏 源码 源码分析 滤镜 瀑布流 灰度 热修复 版本 版本回退 版本控制 状态栏 状态模式 生活 界面 留言板 直方图 相册 相对论 眉县猕猴桃 知识点 码云 磁盘 科学 笔记 策略模式 类图 系统广播 系统概述 系统,发行版, GNU 索引 组件 组合模式 绑定 结构 结构体 编码 编程 网易云信 网格布局 网站广播 网站通知 网络 美化 联合 背景 背景混合 脚手架 膨胀的宇宙 自学Android 自定义 自定义View 自定义插件 色调 蒙版 虚拟 虚拟机 补码 补齐 表单 表达式 装饰模式 西安 观察者模式 规范 视图 视频 视频教程 解耦器模式 设计 设计原则 设计模式 访问者模式 语法 责任链模式 贪吃蛇 走马灯 转换 软件工程 软引用 过滤器 运算符 进程 迭代子模式 适配器 适配器模式 选择器 通信 通道 配置 链表 锐化 错误 键盘 闭包 降噪 陕西地方特产 面向对象 项目优化 项目构建 饱和度 高斯模糊 黑洞
Your browser is out-of-date!

Update your browser to view this website correctly. Update my browser now

×