Intent是应用程序种各个组件联系的桥梁,通信的载体,负责应用程序中数据的传递(运输大队长)
显示的Intent:明确指定要启动的组件,
隐式的Intent:通过IntentFilter来指定要启动的组件
一般在同一个应用中使用显示的Intent,如果跨应用则使用隐式的Intent
如果需要隐式意图启动一个Activity则必须配置CATEGORY_DEFAULT
Android5.0以后不能使用隐式的Intent来启动一个Service,也就是Service组件不能配置IntentFilter
如果通过隐式意图启动的组件不存在,应用将崩溃
if(Intent.resolveActivity(getPackageManager())!=null){
startActivity(sendIntent);
}
如果每次启动都希望选择则:
Intent sendIntent = new Intent(Intent.ACTION_SEND);
String title = getResources().getString(R.string.chooser_title);
Intent chooser = Intent.createChooser(sendIntent,title);
if(sendIntent.resolveActivity(getPackageManager()) != null){
startActivity(chooser);
}
主要用于Intent的延时启动,使用场合有
3.1 包装一个Notification的Intent
3.2 包装一个App Widger的Intent(按Home键启动的Activity)
3.3 包装一个延时启动的Activity(AlarmManager)
通过PendingIntent.getActivity()启动一个activity;
通过PendingIntent.getService()启动一个Service;
通过PendingIntent.getBroadcast()启动一个BroadcastReceiver;
Intent的真正强大之处在于它的隐式Intent,隐式Intent需要配合Intent-filters使用。隐式Intent足够强大,以至于系统提供了大量的Intent方便开发者启动系统应用程序
Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);
intent.setData(Uri.parse("package:" + getPackageName()));
startActivityForResult(intent, APP_SETTING);
Intent intent = new Intent(Settings.ACTION_LOCATION_SOURCE_SETTINGS);
startActivityForResult(intent, GPS_SETTING);
Intent intent = new Intent();
intent.setAction("android.intent.action.VIEW");
intent.addCategory("android.intent.category.DEFAULT");
intent.setDataAndType(Uri.fromFile(newFile("/mnt/sdcard/mobilesafe72_2.apk")),"application/vnd.android.package-archive");
startActivity(intent);
// content从内容提供者中获取数据
// file:从文件中获取数据
Intent intent=new Intent();
intent.setAction(Intent.ACTION_CALL);
intent.setData(Uri.parse("tel:13621310260"));
Intent intent = new Intent();
intent.setAction(Intent.ACTION_VIEW);
intent.putExtra("sms_body",content);
intent.setType("vnd.android-dir/mms-sms");
Intent intent=new Intent();
intent.setAction(Intent.ACTION_SENDTO);
intent.setData(Uri.parse("smsto://0800000132"));
intent.putExtra("sms_body","The SMM text");
Intent intent = new Intent();//intent.setAction(MediaStore.ACTION_VIDEO_CAPTURE);
intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
intent.addCategory(Intent.CATEGORY_DEFAULT);
intent.putExtra(MediaStore.EXTRA_OUTPUT,
FileProvider.getUriForFile(activity, "com.zhijiexing.travel.fileprovider", file));
activity.startActivityForResult(intent, requestcode);
例如现在的栈情况为:A B C D 。D此时通过intent跳转到B,如果这个intent添加FLAG_ACTIVITY_CLEAR_TOP标记,则栈情况变为:A B。如果没有添加这个标记,则栈情况将会变成:A B C D B。也就是说,如果添加了FLAG_ACTIVITY_CLEAR_TOP标记,并且目标Activity在栈中已经存在,则将会把位于该目标activity之上的activity从栈中弹出销毁。这跟上面把B的Launch mode设置成singleTask类似。简而言之,跳转到的activity若已在栈中存在,则将其上的activity都销掉。
例如现在栈1的情况是:A B C。C通过intent跳转到D,并且这个intent添加了FLAG_ACTIVITY_NEW_TASK标记,如果D这个Activity在Manifest.xml中的声明中添加了Task affinity,系统首先会查找有没有和D的Task affinity相同的task栈存在,如果有存在,将D压入那个栈,如果不存在则会新建一个D的affinity的栈将其压入。如果D的Task affinity默认没有设置,则会把其压入栈1,变成:A B C D,这样就和不加FLAG_ACTIVITY_NEW_TASK标记效果是一样的了。注意如果试图从非activity的非正常途径启动一个activity(例见下文“intent.setFlags()方法中参数的用例”),比如从一个service中启动一个activity,则intent比如要添加FLAG_ACTIVITY_NEW_TASK标记(编者按:activity要存在于activity的栈中,而非activity的途径启动activity时必然不存在一个activity的栈,所以要新起一个栈装入启动的activity)。简而言之,跳转到的activity根据情况,可能压在一个新建的栈中。
Intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP);
例如现在栈情况为:A B C。C通过intent跳转到D,这个intent添加FLAG_ACTIVITY_NO_HISTORY标志,则此时界面显示D的内容,但是它并不会压入栈中。如果按返回键,返回到C,栈的情况还是:A B C。如果此时D中又跳转到E,栈的情况变为:A B C E,此时按返回键会回到C,因为D根本就没有被压入栈中。简而言之,跳转到的activity不压在栈中。
和Activity的Launch mode的singleTop类似。如果某个intent添加了这个标志,并且这个intent的目标activity就是栈顶的activity,那么将不会新建一个实例压入栈中。简而言之,目标activity已在栈顶则跳转过去,不在栈顶则在栈顶新建activity。
例如现在栈情况为:A B C,现在要再次启动B,并且设置为该模式,则栈的变化为:A C B
如果这个activity已经启动了,就不产生新的activity,而只是把这个activity实例加到栈顶来就可以了