在程序开发过程当中,常遇到需要启动另一个应用程序的情况,比如在点击软件的一个按钮可以打开地图软件。
如果既有包名又有主类的名字,那就好 办了,
第一种:知道apk的包名和它的主Activity
直接像下面就行:
Intent intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
ComponentName cn = new ComponentName(packageName, className);
intent.setComponent(cn);
startActivity(intent);
Intent intent = new Intent();
PackageManager packageManager = this.getPackageManager();
intent = packageManager.getLaunchIntentForPackage(packageName);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK|Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED | Intent.FLAG_ACTIVITY_CLEAR_TOP) ;
this.startActivity(intent);
or:
// 帮助
private ComponentName help_set;
private final static String help_set_pack = "cn.abc.help";
private final static String help_set_name = "cn.abc.help.MainActivity";
/**
* 启动一个app
* com -- ComponentName 对象,包含apk的包名和主Activity名
* param -- 需要传给apk的参数
*/
private void startApp(ComponentName com, String param) {
if (com != null) {
PackageInfo packageInfo;
try {
packageInfo = getPackageManager().getPackageInfo(com.getPackageName(), 0);
} catch (NameNotFoundException e) {
packageInfo = null;
Toast.makeText(this, "没有安装", Toast.LENGTH_SHORT).show();
e.printStackTrace();
}
try {
Intent intent = new Intent();
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent.setComponent(com);
if (param != null) {
Bundle bundle = new Bundle(); // 创建Bundle对象
bundle.putString("flag", param); // 装入数据
intent.putExtras(bundle); // 把Bundle塞入Intent里面
}
startActivity(intent);
} catch (Exception e) {
Toast.makeText(this, "启动异常", Toast.LENGTH_SHORT).show();
}
}
}
第二种:只知道apk的包名,这种方法最常用了,毕竟要启动的apk不一定是我们自己写的。 /*
* 启动一个app
*/
public void startAPP(String appPackageName){
try{
Intent intent = this.getPackageManager().getLaunchIntentForPackage(appPackageName);
startActivity(intent);
}catch(Exception e){
Toast.makeText(this, "没有安装", Toast.LENGTH_LONG).show();
}
}
综合起来比较第二种方法的代码要简洁得多,比较实用!
注意:
AndroidManifest.xml中需增加
<
uses-permission
android:name
=
“android.permission.GET_TASKS”
/>
一般情况下我们是不知道包名和类名的,那怎么办呢?
其实手机安装的每一个应用的包名都是可以查到的,只要我们查到相关应用的包名,再通过它获得主类名,再按照上面的方法就可以了。
第一步:怎么获得包名?
PackageManager packageManager = context.getPackageManager();
packageManager.getInstalledApplications(0);
这就得到了手机上安装的所有应用的程序信息,当然包括包名,只需要查找自己需要的程序就可以了。
下面给出一个完整的方法,以包名为参数,获取主类名之名再启动应用的具体方法:
public static void openCLD(String packageName,Context context) {
PackageManager packageManager = context.getPackageManager();
PackageInfo pi = null;
try {
pi = packageManager.getPackageInfo("cld.navi.mainframe", 0);
} catch (NameNotFoundException e) {
}
Intent resolveIntent = new Intent(Intent.ACTION_MAIN, null);
resolveIntent.addCategory(Intent.CATEGORY_LAUNCHER);
resolveIntent.setPackage(pi.packageName);
List<ResolveInfo> apps = packageManager.queryIntentActivities(resolveIntent, 0);
ResolveInfo ri = apps.iterator().next();
if (ri != null ) {
String className = ri.activityInfo.name;
Intent intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
ComponentName cn = new ComponentName(packageName, className);
intent.setComponent(cn);
context.startActivity(intent);
}
}
在广播中的页面跳转:
Intent intent = new Intent(context,目标Activity.class); // 这句话可替换成上面
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent );
提示: context 是在广播接收者的
public void onReceive(Context context, Intent intent) 方法中获取的
Context中有一个startActivity方法,Activity继承自Context,重载了startActivity方法。如果使用 Activity的startActivity方法,不会有任何限制,而如果使用Context的startActivity方法的话,就需要开启一个新的task,解决办法是,加一个flag,也就是下面这句
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
如果不添加这句,就会报android.util.AndroidRuntimeException: Calling startActivity() from outside of an Activity,Caused by: android.util.AndroidRuntimeException: Calling startActivity() from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag. Is this really what you want?