广告位联系
返回顶部
分享到

Android在一个app中安装并卸载另一个app的代码

Android 来源:转载 作者:秩名 发布时间:2021-03-13 10:15:14 人浏览
摘要

Android在一个app中安装并卸载另一个app 1.在appsrcmain下新建文件夹asserts,将准备安装的apk文件放在asserts内 2.在appsrcmainres下新建文件夹xml,右击xml文件夹,选择newXML Resource File,File name为文件名,可随意,本文中为filepaths;Root element

Android在一个app中安装并卸载另一个app

1.在app→src→main下新建文件夹asserts,将准备安装的apk文件放在asserts内

2.在app→src→main→res下新建文件夹xml,右击xml文件夹,选择new→XML Resource File,File name为文件名,可随意,本文中为filepaths;Root element为资源类型,输入paths,确定并编辑以下代码:

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
 <external-path name="external_files" path="."/>
</paths>
 

3.编辑AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 package="com.example.myapplication">
 <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
 <application
  android:allowBackup="true"
  android:icon="@mipmap/ic_launcher"
  android:label="@string/app_name"
  android:roundIcon="@mipmap/ic_launcher_round"
  android:supportsRtl="true"
  android:requestLegacyExternalStorage="true"
  android:theme="@style/Theme.MyApplication">
  <provider
   android:name="androidx.core.content.FileProvider"
   android:authorities="包名"
   android:exported="false"
   android:grantUriPermissions="true">
   <meta-data
   android:name="android.support.FILE_PROVIDER_PATHS"
    android:resource="@xml/filepaths步骤2中的文件"
    />
  </provider>
  <activity android:name=".MainActivity">
   <intent-filter>
    <action android:name="android.intent.action.MAIN" />
    <category android:name="android.intent.category.LAUNCHER" />
   </intent-filter>
  </activity>
 </application>
</manifest>
 

4.MainActivity.java

public class MainActivity extends AppCompatActivity {
 Context mContext;
 private Button bt,bt0;
 private TextView tx;
 public static boolean isGrantExternalRW(Activity activity) {
 //权限判定
  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M && activity.checkSelfPermission(
    Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
   activity.requestPermissions(new String[]{
     Manifest.permission.READ_EXTERNAL_STORAGE,
     Manifest.permission.WRITE_EXTERNAL_STORAGE
   }, 1);
   return false;
  }
  return true;
 }
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  mContext = this;
  tx=findViewById(R.id.text);
  bt=findViewById(R.id.button);//安装
  bt0=findViewById(R.id.button0);//卸载
//  Toast.makeText(this, ""+Environment.getExternalStorageDirectory().getAbsolutePath(), 0).show();
  if(!this.isGrantExternalRW(MainActivity.this)){return;}//判断是否有权限修改,这很重要
  else{
  if(copyApkFromAssets(this, "app-debug.apk", Environment.getExternalStorageDirectory().getAbsolutePath()+"/app-debug.apk")){
   bt.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
     new AlertDialog.Builder(mContext)
       .setIcon(R.drawable.ic_launcher)
       .setMessage("是否安装?")
       .setPositiveButton("yes", new DialogInterface.OnClickListener() {
        @Override
        public void onClick(DialogInterface dialog, int which) {
         Intent intent = new Intent(Intent.ACTION_VIEW);
         intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
//         intent.setDataAndType(Uri.parse("file://" + Environment.getExternalStorageDirectory().getAbsolutePath()+"/app-debug.apk"),
//           "application/vnd.android.package-archive");
         File apkFile = new File(Environment.getExternalStorageDirectory()+"/app-debug.apk");
         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
          intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
          Uri uri = FileProvider.getUriForFile(MainActivity.this, "com.example.myapplication", apkFile);
          intent.setDataAndType(uri, "application/vnd.android.package-archive");
         } else {
          intent.setDataAndType(Uri.fromFile(apkFile), "application/vnd.android.package-archive");
         }
         mContext.startActivity(intent);

        }
       }).show();
    }
   });
   bt0.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
     Uri packageURI = Uri.parse("package:com.example.a22");//package:+想要卸载的包名
     Intent intent =new Intent(Intent.ACTION_DELETE);
     intent.setData(packageURI);
     mContext.startActivity(intent);
    }
   });
  }

 }}
 public boolean copyApkFromAssets(Context context, String fileName, String path) {
 //将asserts中的apk文件复制到手机存储中
 //可以从手机存储划到最下面,是单独的文件,没有设置文件夹
  boolean copyIsFinish = false;
  try {
   InputStream is = context.getAssets().open(fileName);
   File file = new File(path);
   file.createNewFile();
   FileOutputStream fos = new FileOutputStream(file);
   int length = is.available();
   byte[] temp = new byte[length];
   int i = 0;
   while ((i = is.read(temp)) > 0) {
    fos.write(temp, 0, i);
   }
   fos.close();
   is.close();

   copyIsFinish = true;
  } catch (IOException e) {
   e.printStackTrace();
  }
  return copyIsFinish;
 }
}

网络上的教程有很多自相矛盾,本人防止权限出错,把能加的权限都设置了一遍,有空可以选择测试一下,应该有的可以删掉。

版权声明 : 本文内容来源于互联网或用户自行发布贡献,该文观点仅代表原作者本人。本站仅提供信息存储空间服务和不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权, 违法违规的内容, 请发送邮件至2530232025#qq.cn(#换@)举报,一经查实,本站将立刻删除。
原文链接 : https://blog.csdn.net/qq_42024030/article/details/114640004
相关文章
  • Kotlin的Collection与Sequence操作异同点介绍

    Kotlin的Collection与Sequence操作异同点介绍
    在Android开发中,集合是我们必备的容器,Kotlin的标准库中提供了很多处理集合的方法,而且还提供了两种基于容器的工作方式:Collection 和
  • 实现一个Kotlin函数类型方法

    实现一个Kotlin函数类型方法
    接口与函数类型 业务开发中,经常会有实现一个函数式接口(即接口只有一个方法需要实现)的场景,大家应该都会不假思索的写出如下代
  • Android10 App启动Activity源码分析
    ActivityThread的main方法 让我们把目光聚焦到ActivityThread的main方法上。 ActivityThread的源码路径为/frameworks/base/core/java/android/app/ActivityThread。 1 2
  • Android10客户端事务管理ClientLifecycleManager源码解析

    Android10客户端事务管理ClientLifecycleManager源码解析
    在Android 10 App启动分析之Activity启动篇(二)一文中,简单地介绍了Activity的生命周期管理器是如何调度Activity进入onCreate生命周期的流程。这
  • Kotlin对象的懒加载方式by lazy与lateinit异同介绍

    Kotlin对象的懒加载方式by lazy与lateinit异同介绍
    属性或对象的延时加载是我们相当常用的,一般我们都是使用 lateinit 和 by lazy 来实现。 他们两者都是延时初始化,那么在使用时那么他们两
  • Android类加载流程分析

    Android类加载流程分析
    本文分析的代码基于Android8.1.0源码。 流程分析 从loadClass开始,我们来看下Android中类加载的流程 /libcore/ojluni/src/main/java/java/lang/ClassLoader.ja
  • Android实现读写USB串口数据的代码

    Android实现读写USB串口数据的代码
    最近在研究USB方面的内容;先后做了关于Android读写HID、串口设备的DEMO。本文比较简单,主要介绍的是Android实现读取串口数据的功能 废话不
  • Epoxy - 在RecyclerView中构建复杂界面
    Diffing 对于复杂数据结构支持的多个视图类型展示在屏幕上, Epoxy此时是尤其有用的. 在这些场景中, 数据可能会被网络请求, 异步 Observable, 用
  • Android性能优化的详细介绍

    Android性能优化的详细介绍
    性能优化是一个app很重要的一部分,一个性能优良的app从被下载到启动到使用都能给用户到来很好的体验。自然我们做性能优化也是从被下
  • Android进阶宝典-插件化2(Hook启动插件中四大组件

    Android进阶宝典-插件化2(Hook启动插件中四大组件
    在上一节,我们主要介绍了如果通过反射来加载插件中的类,调用类中的方法;既然插件是一个apk,其实最重要的是启动插件中的Activity、
  • 本站所有内容来源于互联网或用户自行发布,本站仅提供信息存储空间服务,不拥有版权,不承担法律责任。如有侵犯您的权益,请您联系站长处理!
  • Copyright © 2017-2022 F11.CN All Rights Reserved. F11站长开发者网 版权所有 | 苏ICP备2022031554号-1 | 51LA统计