Android应用程序的编译和打包

Android系统的APK应用程序可以有以下几种编译方式 

借助系统编译:利用Android.mk 文件将众多小项目组织起来

借助IDE编译:AndroidStudio

命令行编译 : 比如利用gradle脚本编译APK应用。 

一、 通过命令行编译和打包APK

编译命令(Window系统)

./gradlew  build   检查依赖并且编译打包

./gradlew assembleDebug/assembleRelease 编译并打出Debug包 /Release包

./gradlew installDebug 编译出Debug包并且安装

./gradlew installRelease 编译出Release包并且安装

./gradlew assembleDebug/assembleRelease  --info     编译包并且打印日志 

./gradlew assembleDebug/assembleRelease  --scan  编译并且输出更详细的报告 

./gradlew clean  清除构建目录下的文件

./gradlew  unsinstall <packageName>   卸载安装 

./gradlew  uninstallDebug / uninstallRelease   卸载安装包 

// task相关 

./gradlew --tasks  查看主要Task

./gradlew tasks --all  查看所有Task 

./gradlew <taskName>  or  ./gradlew   :<moduleName>:<taskName>  执行Task 

// 查看依赖 

./gradlew dependencies   查看项目根目录下的依赖 

./gradlew  :app:dependencies   查看app模块下的依赖

./gradlew  :app:dependencies  >  dependencies.txt  查看依赖输出到文件 

//性能相关

./gradlew assembleDebug  --offline  离线编译 

./gradlew  assembleDebug --build-cache  可开启缓存

./gradlew  assembleDebug --no-build-cache  不开启

./gradlew  assembleDebug --configuration-cache   配置缓存开启

./gradlew  assembleDebug --no-configuration-cache  不开启

./gradlew  assembleDebug  --parallel  并行构建开启 

./gradlew  assembleDebug   --no-parallel 不开启  

二、 APK编译过程详情

  以apk后缀的文件是Android应用系统的标准格式,其实是一个压缩包,用压缩工具截出来,可以看到一个apk应用程序至少包含一下几部分内容 : 

|-- AndroidManifest.xml 
|-- cleasses.dex
|-- resources.arsc
|-- res
|   |-- drawable
|   |   -- *.png  
|   |-- layout
|       -- *.xml 
|
|-- META-INF
|   |-- CERT.RSA
|   |-- CERT.SF]
|   |-- MANIFEST.MF
|   |
|

APK编译过程图解 

 

 可以清楚看到整个编译过程涉及到了许多工具

1. 首先是.aidl文件要经过aidl工具转换成编译器能处理的Java接口文件。 

2. 同时资源文件将aapt处理为最终的resources.arsc,并且生产R.java。

3. Java编译器统一将Java源码,R.java等文件统一变异成.class 文件。 

4. dex工具将.class工具转换成Android系统所能识别的格式dex。

5. 接下里dex、资源等通过apkbuilder生成初始的APK 。

6. apk签名。 签名工具有很多。 

7. 签名后apk通过zipalign进行优化。

三、 应用程序签名源码简析

接下里分析一下源码来了解META-INF文件的用途以及签名校验大致流程。 应用程序安装时候,首先是PackageManger会对其进行初始化。这包含对签名和文件哈希值的检查。 

1.PackageParser 负责解析应用程序包,并完成安全校验

 private void installPackageTracedLI(InstallArgs args, PackageInstalledInfo res) {....  PackageParser pp = new PackageParser();...pkg = pp.parsePackage(tmpPackageFile, parseFlags);.... try {// either use what we've been given or parse directly from the APKif (args.certificates != null) {try {PackageParser.populateCertificates(pkg, args.certificates);} catch (PackageParserException e) {// there was something wrong with the certificates we were given;// try to pull them from the APKPackageParser.collectCertificates(pkg, parseFlags);}} else {PackageParser.collectCertificates(pkg, parseFlags);}} catch (PackageParserException e) {res.setError("Failed collect during installPackageLI", e);return;}
}

2. PackageParser通过collectCertificates()来检验程序包中所有文件是否都符合要求,然后才能返回PackageManager继续执行安装过程。 这个函数分为以下关键部分: 

 解析package包,得到所以文件 ,对这些文件逐一检验。 安全检查关键在于loadCertificates

     private static Certificate[][] loadCertificates(StrictJarFile jarFile, ZipEntry entry)
633              throws PackageParserException {
634          InputStream is = null;
635          try {
636              // We must read the stream for the JarEntry to retrieve
637              // its certificates.
638              is = jarFile.getInputStream(entry);
639              readFullyIgnoringContents(is);
640              return jarFile.getCertificateChains(entry);
641          } catch (IOException | RuntimeException e) {
642              throw new PackageParserException(INSTALL_PARSE_FAILED_UNEXPECTED_EXCEPTION,
643                      "Failed reading " + entry.getName() + " in " + jarFile, e);
644          } finally {
645              IoUtils.closeQuietly(is);
646          }
647      }
648  

整个检验过程主要涉及以下几点: 

CERT.RSA

这是应用程序开发者提供的证书,包含开发者的公钥和一些列身份信息。因为是自签名的,就不需要CA认证。 

CERT.SF 

后缀名.SF 应该是Signature File的缩写,所以就是我们所说的签名文件。 它是对某个文件的HASH值进行私钥加密产生的。 它是对真个MANIFEST.MF 进行加密---它包好了应用中所有文件的Hash值。 

MANIFEST.MF 

 /frameworks/base/core/java/android/util/jar/StrictJarVerifier.java

    void verify() {
141              byte[] d = digest.digest();
142              if (!MessageDigest.isEqual(d, Base64.decode(hash))) {
143                  throw invalidDigest(JarFile.MANIFEST_NAME, name, name);
144              }
145              verifiedEntries.put(name, certChains);
146          }

文件 SHA-1是经过了Base64编码的,所以这里要解码在对比。 

这样就确保了Android的应用程序没有被恶意篡改,有效的保护了开发者的权益。  

V2的校验过程(略)

四、应用签名

我们可以利用sdk中的build-tools里面的apksigner工具来验证一下apk用了那种方式签名。  

 

V2签名的信息存储在Android应用程序的APK文件中的签名块(Signature Block)中。签名块是APK文件的一部分,用于存储应用程序的数字签名信息。

在APK文件的二进制结构中,签名块位于文件的末尾,紧跟在ZIP目录结构之后。签名块使用特定的格式存储签名信息,包括应用程序的证书和签名本身。

要查看APK文件中的签名信息,你可以使用Android SDK提供的工具和命令行选项。一个常用的工具是apksigner,它可以用于验证应用程序的签名并提供签名块的详细信息。

以下是使用apksigner命令查看APK文件签名信息的示例:

apksigner verify --print-certs path/to/your/app.apk

这些签名方案用于验证Android应用程序的完整性和真实性,并确保应用程序在安装和运行时没有被篡改。

V1签名(JAR签名):V1签名是早期版本的Android应用程序签名方案。它使用基于JAR(Java Archive)的签名方式,将整个APK文件视为JAR文件,并使用基于RSA的数字签名算法对其进行签名。V1签名方案在Android 1.0至7.0版本中使用。

V2签名(APK签名):V2签名是Android引入的第二个签名方案,用于增强应用程序的安全性和完整性保护。V2签名将APK文件分为多个部分,并对每个部分进行签名,以提供更好的验证和错误检测。V2签名方案使用基于RSA的数字签名算法,并在APK文件末尾添加签名块。V2签名方案从Android 7.0开始引入,并成为默认的签名方案。

V3签名(APK签名):V3签名是在V2签名基础上进行改进的签名方案。它引入了APK签名方案V2的补充,并使用ECDSA(椭圆曲线数字签名算法)来生成签名。V3签名进一步提高了应用程序验证的安全性,并提供了更好的完整性保护。V3签名方案从Android 9.0开始引入。

V4签名(APK签名):V4签名是Android在Android 11中引入的新签名方案。它是对V2签名的扩展,旨在提供更快的应用程序安装速度和更小的APK文件大小。V4签名使用了更高效的签名算法和压缩技术,以减少签名数据的大小。V4签名方案需要使用Android 11及更高版本的构建工具和设备支持。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/671177.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

没有联合和枚举 , C语言怎么能在江湖混 ?

本篇会加入个人的所谓‘鱼式疯言’ ❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言 而是理解过并总结出来通俗易懂的大白话, 我会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的. &#x1f92d;&#x1f92d;&#x1f92d;可能说的不是那么严谨.但小编初心是能让更多人能…

探索C语言结构体:编程中的利器与艺术

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;C语言学习 贝蒂的主页&#xff1a;Betty‘s blog 1. 常量与变量 1. 什么是结构体 在C语言中本身就自带了一些数据类型&#x…

LLMs之miqu-1-70b:miqu-1-70b的简介、安装和使用方法、案例应用之详细攻略

LLMs之miqu-1-70b&#xff1a;miqu-1-70b的简介、安装和使用方法、案例应用之详细攻略 目录 miqu-1-70b的简介 miqu-1-70b的安装和使用方法 1、安装 2、使用方法 miqu-1-70b的案例应用 miqu-1-70b的简介 2024年1月28日&#xff0c;发布了miqu 70b&#xff0c;潜在系列中的…

Linux系统调试课:ftrace跟踪器介绍

文章目录 一、什么是frace跟踪器?二、Ftrace 配置三、Ftrace 文件系统四、Ftrace 初体验五、函数跟踪六、Ftrace function_graph七、函数 Profiler沉淀、分享、成长,让自己和他人都能有所收获!😄 一、什么是frace跟踪器? 操作系统内核对应用开发工程师来说就像一个黑盒,…

elementUI 表格中如何合并动态数据的单元格

elementUI 表格中如何合并动态数据的单元格 ui中提供的案例是固定写法无法满足 实际开发需求 下面进行改造如下 准备数据如下 //在表格中 设置单元格的方法 :span-method"spanMethodFun" <el-table :data"tableData" border :span-method"spa…

手撕spring bean的加载过程

这里我们采用手撕源码的方式&#xff0c;开始探索spring boot源码中最有意思的部分-bean的生命周期&#xff0c;也可以通过其中的原理理解很多面试以及工作中偶发遇到的问题。 springboot基于约定大于配置的思想对spring进行优化&#xff0c;使得这个框架变得更加轻量化&#…

Backtrader 文档学习- Observers

Backtrader 文档学习- Observers 1.概述 在backtrader中运行的策略主要处理数据源和指标。 数据源被加载到Cerebro实例中&#xff0c;并最终成为策略的一部分&#xff08;解析和提供实例的属性&#xff09;&#xff0c;而指标则由策略本身声明和管理。 到目前为止&#xff0c…

LabVIEW多功能接口卡驱动

LabVIEW多功能接口卡驱动 随着自动化测试系统的复杂性增加&#xff0c;对数据采集与处理的需求不断提高。研究基于LabVIEW开发平台&#xff0c;实现对一种通用多功能接口卡的驱动&#xff0c;以支持多通道数据采集及处理功能&#xff0c;展现LabVIEW在自动化和测量领域的强大能…

如何部署Docker Registry并实现无公网ip远程连接本地镜像仓库

文章目录 1. 部署Docker Registry2. 本地测试推送镜像3. Linux 安装cpolar4. 配置Docker Registry公网访问地址5. 公网远程推送Docker Registry6. 固定Docker Registry公网地址 Docker Registry 本地镜像仓库,简单几步结合cpolar内网穿透工具实现远程pull or push (拉取和推送)…

IPv4的公网地址不够?NAT机制可能是当下最好的解决方案

目录 1.前言 2.介绍 3.NAT机制详解 1.前言 我们都知道IPv4的地址范围是32个字节,这其中还有很多地址是不可用的.比如127.*,这些都是环回地址.那么在网路发展日新月异的今天,互联网设备越来越多,我们该如何解决IP地址不够用的问题呢?目前有一种主流的解决方案,也是大家都在用…

Visual Studio 2022 查看类关系图

这里写自定义目录标题 右键要查看的项目 -“查看”-“查看类图”效果展示&#xff1a; 原文地址 www.cnblogs.com 步骤1&#xff1a;勾选扩展开发 步骤2: 勾选类设计器 右键要查看的项目 -“查看”-“查看类图” 效果展示&#xff1a;

操作系统-02-Ubuntu 常用命令等汇总

Ubuntu Ubuntu是一个开源软件平台&#xff0c;可以在从智能手机、平板电脑和个人电脑到服务器和云端等各种设备上运行。 ubuntu Title: Desktop Ubuntu desktop->command: CtrlAltF2~F6 command->desktop: CtrlAltF7 Login Change to root user $ sudu suLogin to al…

AI嵌入式K210项目(27)-条形码识别

文章目录 前言一、什么是条形码&#xff1f;二、实验准备三、实验过程四、API接口总结 前言 本章介绍基于机器视觉实现条形码识别&#xff0c;主要包含两个过程&#xff0c;首先检测图像中是否有条形码&#xff0c;如果有则框出并打印条形码信息&#xff1b; 一、什么是条形码…

ShardingSphere实现openGauss分布式架构

本文档采用openGauss结合ShardingSphere中间件的架构&#xff0c;实现openGauss数据库分布式OLAP场景的环境部署。 术语说明&#xff1a; 开源数据库引擎&#xff1a;openGauss shardingsphere Proxy&#xff1a;定位为透明化的数据库代理端&#xff0c;提供封装了数据库二进…

示波器的电压怎么和测量的电压不一样,空比增加以后 示波器显示的电压反而降低了?

这两天在调电机&#xff0c;发现我用pwm信号&#xff0c;发送给示波器&#xff0c;但是示波器测量的电压和万用表测量的完全不一样&#xff0c;而有奇怪的现象&#xff0c;空比增加以后 示波器显示的电压反而降低了。 最后发现是通道没有设置&#xff0c;要设置成直流模式 点ch…

docker之程序镜像的制作

目录 一、每种资源的预安装&#xff08;基础&#xff09; 安装 nginx安装 redis 二、dockerfile文件制作&#xff08;基础&#xff09; 打包 redis 镜像 创建镜像制作空间制作dockerfile 打包 nginx 镜像 三、创建组合镜像&#xff08;方式一&#xff09; 生成centos容器并…

补题:leetcode第382场周赛 3022. 给定操作次数内使剩余元素的或值最小

3022. 给定操作次数内使剩余元素的或值最小 - 力扣&#xff08;LeetCode&#xff09; 拆位 n个数进行或运算的结果最小&#xff0c;每次操作可以对相邻的两个数进行与运算&#xff0c;至多进行k次操作 n个数进行或运算&#xff0c;可以对每个数进行拆解&#xff0c;拆解成最小…

分享springboot框架的一个开源的本地开发部署教程(若依开源项目开发部署过程分享持续更新二开宝藏项目PostgresSQL数据库版)

1首先介绍下若依项目&#xff1a; 若依是一个基于Spring Boot和Spring Cloud技术栈开发的多租户权限管理系统。该开源项目提供了一套完整的权限管理解决方案&#xff0c;包括用户管理、角色管理、菜单管理、部门管理、岗位管理等功能。 若依项目采用前后端分离的架构&#xf…

使用代理IP有风险吗?如何安全使用代理IP?

代理IP用途无处不在。它们允许您隐藏真实IP地址&#xff0c;从而实现匿名性和隐私保护。这对于保护个人信息、绕过地理受限的内容或访问特定网站都至关重要。 然而&#xff0c;正如任何技术工具一样&#xff0c;代理IP地址也伴随着潜在的风险和威胁。不法分子可能会滥用代理IP…

ue5 蓝图内修改组件附加的插槽

Target是目标,Parent是要加到哪个骨骼上,socketName是骨骼对应的插槽