Android-PackageManager与startActivity启动第三方应用的Activity

参考:

Android PackageManager 详解
Android应用间跳转
Android随笔之——PackageManager详解

Android系统为我们提供了很多服务管理的类,包括ActivityManager、PowerManager(电源管理)、AudioManager(音频管理)等。除此之外,还提供了一个PackageManger管理类,它的主要职责是管理应用程序包。 通过PackageManager,我们就可以获取应用程序信息。

  提到PackageManager,就得提一下AndroidManifest.XML文件了。AndroidManifest.xml是Android应用程序中最重要的文件之一。它是Android程序的全局配置文件,是每个 android程序中必须的文件。它位于我们开发的应用程序的根目录下,描述了package中的全局数据,包括package中暴露的组件 (activities, services, 等等),以及他们各自的实现类,各种能被处理的数据和启动位置等重要信息。
  因此,该文件提供了Android系统所需要的关于该应用程序的必要信息,即在该应用程序的任何代码运行之前系统所必须拥有的信息。

  PackageManager获取的信息即来自AndroidManifest.XML。为了便于理解,从网上找了一张AnroidManifest.xml文件节点说明图:

PackageManager的功能:
1、安装,卸载应用
2、查询permission相关信息
3、查询Application相关信息(application,activity,receiver,service,provider及相应属性等)
4、查询已安装应用
5、增加,删除permission
6、清除用户数据、缓存,代码段等

PackageManager相关类和方法介绍:

1、PackageManager类

说明: 获得已安装的应用程序信息 。可以通过getPackageManager()方法获得。 

常用方法: 
public abstract PackageManager getPackageManager()  
功能:获得一个PackageManger对象  

public abstract Drawable getApplicationIcon(String packageName)
参数: packageName 包名
功能:返回给定包名的图标,否则返回null

public abstract ApplicationInfo getApplicationInfo(String packageName, int flags)
参数:
  packagename 包名
  flags 该ApplicationInfo是此flags标记,通常可以直接赋予常数0即可
功能:返回该ApplicationInfo对象

public abstract List<ApplicationInfo> getInstalledApplications(int flags)
参数:
  flag为一般为GET_UNINSTALLED_PACKAGES,那么此时会返回所有ApplicationInfo。我们可以对ApplicationInfo
  的flags过滤,得到我们需要的。
功能:返回给定条件的所有PackageInfo

public abstract List<PackageInfo> getInstalledPackages(int flags) 
参数如上
功能:返回给定条件的所有PackageInfo

public abstract ResolveInfo resolveActivity(Intent intent, int flags)
参数:  
  intent 查寻条件,Activity所配置的action和category
  flags: MATCH_DEFAULT_ONLY     :Category必须带有CATEGORY_DEFAULT的Activity,才匹配
       GET_INTENT_FILTERS     :匹配Intent条件即可
       GET_RESOLVED_FILTER    :匹配Intent条件即可
功能 :返回给定条件的ResolveInfo对象(本质上是Activity)

public abstract List<ResolveInfo> queryIntentActivities(Intent intent, int flags)
参数同上
功能 :返回给定条件的所有ResolveInfo对象(本质上是Activity),集合对象

public abstract ResolveInfo resolveService(Intent intent, int flags)
参数同上
功能 :返回给定条件的ResolveInfo对象(本质上是Service)

public abstract List<ResolveInfo> queryIntentServices(Intent intent, int flags)
参数同上
功能 :返回给定条件的所有ResolveInfo对象(本质上是Service),集合对象

2、PackageItemInfo类

说明: AndroidManifest.xml文件中所有节点的基类,提供了这些节点的基本信息:label、icon、 meta-data
它并不直接使用,而是由子类继承然后调用相应方法。

3、ApplicationInfo类 继承自 PackageItemInfo类

说明:获取一个特定引用程序中<application>节点的信息。

字段说明:
        flags字段: FLAG_SYSTEM 系统应用程序
                    FLAG_EXTERNAL_STORAGE 表示该应用安装在sdcard中

常用方法继承至PackageItemInfo类中的loadIcon()和loadLabel()

String appName = resolveInfo.loadLabel(pm).toString();// 获取应用名称
String packageName = resolveInfo.activityInfo.packageName;// 包名
String className = resolveInfo.activityInfo.name;// 入口类名

4、ActivityInfo类 继承自 PackageItemInfo类

说明: 获得应用程序中<activity/>或者 <receiver/>节点的信息 。我们可以通过它来获取我们设置的任何属性,包括theme 、launchMode、launchmode等

常用方法继承至PackageItemInfo类中的loadIcon()和loadLabel()

5、ServiceInfo类 继承自 PackageItemInfo类

说明:与ActivityInfo类似,代表<service>节点信息

6、ResolveInfo类

说明:根据<intent>节点来获取其上一层目录的信息,通常是<activity><receiver><service>节点信息。
常用方法有loadIcon(PackageManager pm)和loadLabel(PackageManager pm)

java代码:

package com.example.administrator.mydb;

import android.app.Activity;
import android.content.ComponentName;
import android.content.Intent;
import android.content.pm.ApplicationInfo;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends Activity {

    private Intent intent = null;
    private TextView textView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        textView = ((TextView) this.findViewById(R.id.tv));
        loadApps();

        /*跳转微信界面*/
        ((Button) this.findViewById(R.id.btn_1)).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                startActivity(getPackageManager().getLaunchIntentForPackage("com.tencent.mm"));
            }
        });

        /*跳转其他应用,指定包名,不指定类名*/
        ((Button) this.findViewById(R.id.btn_main)).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //不知道类名,跳转到指定的包
                PackageManager pm = getPackageManager();
                PackageInfo pi = null;
                try {
                    pi = pm.getPackageInfo("com.example.administrator.downloadimgdemo", 0);
                } catch (PackageManager.NameNotFoundException e) {
                    e.printStackTrace();
                }

                Intent resolveIntent = new Intent(Intent.ACTION_MAIN, null);
                resolveIntent.addCategory(Intent.CATEGORY_LAUNCHER);
                resolveIntent.setPackage(pi.packageName);

                //找到匹配intent的所有Activity,这个方法可以在intent跳转前调用用来检测是否有符合条件的activity
                List<ResolveInfo> apps = pm.queryIntentActivities(resolveIntent, 0);
                ResolveInfo ri = apps.iterator().next();
                if (ri != null) {
                    String className = ri.activityInfo.name;

                    intent = new Intent(Intent.ACTION_MAIN);
                    intent.addCategory(Intent.CATEGORY_LAUNCHER);

                    ComponentName cn = new ComponentName("com.example.administrator.downloadimgdemo", className);

                    intent.setComponent(cn);
                }
            }
        });


        /*跳转其他应用,指定类名*/
        ((Button) this.findViewById(R.id.btn_jump)).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //知道要跳转应用的包名、类名
                ComponentName comp = new ComponentName("com.example.administrator.downloadimgdemo",
                        "com.example.administrator.downloadimgdemo.GetAllImg");
                final Intent intent = new Intent();
                intent.setComponent(comp);
                startActivity(intent);
            }
        });
    }


    // 列出普通应用程序
    private void loadApps() {
        List<PackageInfo> customApps = new ArrayList<PackageInfo>(); // 普通应用程序列表
        List<PackageInfo> systemApps = new ArrayList<PackageInfo>(); // 系统应用程序列表
        // 得到PackageManager对象
        PackageManager pm = this.getPackageManager();
        // 得到系统安装的所有程序包的PackageInfo对象
        List<PackageInfo> packages = pm.getInstalledPackages(0);
        StringBuffer sb = new StringBuffer();
        for (PackageInfo pi : packages) {
            // 列出普通应用
            if ((pi.applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) <= 0) {
                customApps.add(pi);
                sb.append("》》》》普通pi======" + pi + "\n");
            }
            // 列出系统应用,总是感觉这里设计的有问题,希望高手指点
            if ((pi.applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) > 0) {
                systemApps.add(pi);
                sb.append("《《《《系统pi======" + pi + "\n");
            }
        }
        textView.setText(sb.toString());
    }


}

布局:

<?xml version="1.0" encoding="utf-8"?>
<ScrollView
        xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        android:id="@+id/activity_main"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:paddingLeft="@dimen/activity_horizontal_margin"
        android:paddingRight="@dimen/activity_horizontal_margin"
        android:paddingTop="@dimen/activity_vertical_margin"
        android:paddingBottom="@dimen/activity_vertical_margin"
        tools:context="com.example.administrator.mydb.MainActivity">

    <LinearLayout android:layout_width="match_parent"
                  android:layout_height="match_parent"
                  android:orientation="vertical">


        <Button
                android:id="@+id/btn_jump"
                android:text="跳转其他应用(不是启动页面的activity)"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"/>

        <Button
                android:id="@+id/btn_main"
                android:text="跳转其他应用(启动页面的activity)"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"/>

        <Button
                android:id="@+id/btn_1"
                android:text="跳转微信"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"/>

        <TextView
                android:id="@+id/tv"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="Hello World!"/>
    </LinearLayout>
</ScrollView>

遇到问题:

java.lang.SecurityException: Permission Denial: starting Intent { act=android.intent.action.VIEW cmp=.SnsTimeLineUI } from …… not exported from uid 10092….

参考:

通过startActivity启动第三方应用的Activity时崩溃Permission Denial(比如打开微信朋友圈)

问题分析及解决:

只有当Activity向外声明了自己是可以处理某些intent action时,第三方app才能通过intent去启动它。通过配置文件中的intent过滤器来声明:

<activity android:name=".GetAllImg">
    <intent-filter >
        <action android:name="android.intent.action.VIEW"/>
    </intent-filter>
</activity>

当知道了package name和class name,可以这样去启动第三方的activity:

ComponentName comp = new ComponentName("com.example.administrator.downloadimgdemo","com.example.administrator.downloadimgdemo.GetAllImg");
                Intent intent = new Intent();
                intent.setComponent(comp);
                startActivity(intent);

这样做是没问题的。但假如没有在manifest中声明,startActivity将导致崩溃。

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android 6.0及以上版本中,需要动态申请权限才能访问存储空间。以下是一个示例代码,用于启动第三方应用并请求存储权限: ```java private static final int REQUEST_CODE_STORAGE_PERMISSION = 1; // 启动第三方应用 private void launchThirdPartyApp() { Intent launchIntent = getPackageManager().getLaunchIntentForPackage("com.example.thirdpartyapp"); if (launchIntent != null) { if (checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED) { // 如果已经授权,直接启动第三方应用 startActivity(launchIntent); } else { // 如果没有授权,请求存储权限 requestPermissions(new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, REQUEST_CODE_STORAGE_PERMISSION); } } else { // 第三方应用未安装 Toast.makeText(this, "Third party app not installed.", Toast.LENGTH_SHORT).show(); } } // 处理权限请求结果 @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { if (requestCode == REQUEST_CODE_STORAGE_PERMISSION) { if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { // 用户已授权,启动第三方应用 Intent launchIntent = getPackageManager().getLaunchIntentForPackage("com.example.thirdpartyapp"); startActivity(launchIntent); } else { // 用户拒绝授权,无法启动第三方应用 Toast.makeText(this, "Permission denied, cannot launch third party app.", Toast.LENGTH_SHORT).show(); } } } ``` 在上面的代码中,`launchThirdPartyApp()` 方法会尝试启动指定包名的第三方应用。如果当前应用已经被授予存储权限,直接启动第三方应用;否则,请求存储权限,并在 `onRequestPermissionsResult()` 方法中处理用户的授权结果。如果用户授权,再次尝试启动第三方应用;如果用户拒绝授权,提示用户无法启动第三方应用

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值