android菜鸟学习笔记11----Intent的两点补充
关于Intent的两点补充:
1.隐式Intent启动组件,会有一个Intent解析的过程,若找不到能够处理该Intent的组件,程序就会异常终止。一个合理的做法是,在使用Intent实例启动组件如:startActivity(intent)之前,最好能判断一下该调用能否解析为一个Activity。为了实现该功能,Intent提供了一个方法:
ComponentName resolveActivity(PackageManager pm) :该方法接收一个包管理器对象作为参数,通过查找该包管理器,返回能够处理该Intent的Activity的Component对象,没有找到能处理该Inent的组件时则返回null。
Intent intent1 = new Intent(); intent1.setAction(Intent.ACTION_VIEW); intent1.setType("text/html"); Log.i("tag",intent1.toString()); ComponentName resolveActivity = intent1.resolveActivity(getPackageManager()); Log.i("tag",resolveActivity.getClassName()); Log.i("tag",resolveActivity.getPackageName()); Log.i("tag",resolveActivity.toString()); startActivity(intent1);
NormalActivity中配置<intent-filter>
<activity android:name=".NormalActivity"> <intent-filter > <action android:name="android.intent.action.VIEW"/> <category android:name="android.intent.category.DEFAULT"/> <data android:mimeType="text/html"/> </intent-filter> </activity>
运行信息:
可见解析到cn.csc.lifecycle.NormalActivity能够处理该Intent。
若找不到能够处理该Intent的组件时,会返回null,所以,安全的启动Activity的做法是:
if(intent1.resolveActivity(getPackageManager())!=null){ startActivity(intent1); }
当然,可以加上找不到能处理intent1的组件的提示信息等。
注意,若用多个组件都能处理该Intent实例时,如:
Intent intent1 = new Intent(); intent1.setAction(Intent.ACTION_VIEW); intent1.setData(Uri.parse("http://www.baidu.com")); Log.i("tag",intent1.toString()); ComponentName resolveActivity = intent1.resolveActivity(getPackageManager()); Log.i("tag",resolveActivity.toString()); startActivity(intent1);
<activity android:name=".NormalActivity"> <intent-filter > <action android:name="android.intent.action.VIEW"/> <category android:name="android.intent.category.DEFAULT"/> <data android:scheme="http"/> </intent-filter> </activity>
此时,浏览器和NormalActivity都能处理该Intent实例。
运行效果:
返回的Component信息为:
这个ResolverActivity其实就是一个选择打开程序的对话框,当有多个组件能处理Intent实例,但是没有设置默认的处理程序时,就会交由ResolverActivty来处理,而ResolverActivity的做法就是把所有能处理的都显示在一个列表中,由用户决定采用哪个来处理。当我们勾选了下面的use by default for this action时,之后选择的组件将会作为该Intent的默认处理程序,该设置会保存在手机或者模拟器中,下次在出现该Intent时,直接使用设置的默认组件去处理,而不再转到ResolverActivity。
如,这里我选择了Browser作为默认处理程序,下次再启动改程序,输出的运行信息:
2.关于使用Intent在Activity之间传递数据,在之前都是传递些内置类型的数据,若想传递自定义类型的数据,应该怎么办呢?
注意到在Intent中有下面两对方法:
Serializable getSerializableExtra(String name)
Intent putExtra(String name, Serializable value)
放入/获取Serializable类型的值。
<T extends Parcelable> T getParcelableExtra(String name)
Intent putExtra(String name, Parcelable value)
放入/获取Parcelable类型的值。
Serializable和Parcelable都是接口,即要想使用Intent传递自定义类型的数据,则自定义类型的数据应当事先Serializable接口或者Parcelable接口。
Serializable接口:
java.io.Serializable是一个空的起到标记作用的接口。标记实现该接口,只需要在类定义时声明实现该接口即可。实现该接口的类表明能够被序列化和反序列化,对大多数的类来说已然足够了,当然,若想更细粒度的操纵序列化过程,可以实现下面这两个方法:
private void writeObject(java.io.ObjectOutputStream out) throws IOException
private void readObject(java.io.ObjectInputStream in) throws IOException, ClassNotFoundException
一般情况下,都用不到的。
这里简单声明实现该接口,如:
import java.io.Serializable; public class MyData implements Serializable { private String stringData; private int intData; public MyData(String stringData, int intData) { super(); this.stringData = stringData; this.intData = intData; } public String getStringData() { return stringData; } public void setStringData(String stringData) { this.stringData = stringData; } public int getIntData() { return intData; } public void setIntData(int intData) { this.intData = intData; } }
之后,便可以传递MyData类型数据了:
MyData data1 = new MyData(“aaa”,111); Intent intent1 = new Intent(); intent1.putExtra(“mydata”, data1);
取得该数据:
MyData data1 = (MyData)getIntent.getSerializableExtra(“data1”);
Parcelable接口:相比Serializable要复杂一点,需要实现里面的抽象方法。
android.os.Parcelable接口,帮助文档中,有一个典型的实现例子如下,可以用作参考,自己实现该接口时,模仿即可:
public class MyParcelable implements Parcelable { private int mData; public int describeContents() { return 0; } public void writeToParcel(Parcel out, int flags) { out.writeInt(mData); } public static final Parcelable.Creator<MyParcelable> CREATOR = new Parcelable.Creator<MyParcelable>() { public MyParcelable createFromParcel(Parcel in) { return new MyParcelable(in); } public MyParcelable[] newArray(int size) { return new MyParcelable[size]; } }; private MyParcelable(Parcel in) { mData = in.readInt(); } }
定义各种内置类型的数据,然后在writeToParcel分别根据不同的类型调用对应类型的
out.writeXXX(mData);在构造方法中,分别调用in.readXX()即可。
然后就可以调用Intent putExtra(String name, Parcelable value)去放入该自定义类型数据,调用<T extends Parcelable> T getParcelableExtra(String name)获取该自定义类型数据。
最新文章
- [LeetCode] Is Subsequence 是子序列
- java开发环境搭建
- VMware Workstation下安装centos提示硬盘不支持的解决方法
- KeepAlived主备/主主模型高可用Nginx
- 编程语言java-并发(锁)
- hibernate中session,HQL,持久化,一对多,多对一
- 应用程序域 z
- Oracle入门学习笔记
- Pie(求最小身高差,dp)
- 使用jQuery快速高效制作网页特效-----------------------------之jQuery事件与动画
- vue-cli创建自己的项目
- Android 软键盘的显示和隐藏,这样操作就对了
- 微信小程序样式旋转
- 一种hyperscan API使用(1)
- 修改select默认小箭头
- TCP/IP 笔记 - 安全
- 统计分析与R软件-chapter2-5
- 使用loki+ mtail + grafana + prometheus server分析应用问题
- DES加密Java实现
- STM8S——8位基本定时器(TIM4)