当前位置:   article > 正文

编写Android.mk把Android studio项目编译到AOSP源码中_androidstudio项目编译进源码中

androidstudio项目编译进源码中

前言:

             在工作,我们利用Android studio开发apk是非常方便的,当我们要把工程代码放在android

源码中编译的时候,需要我们自己编写Andorid.mk 文件。以下内容是对Android.mk文件中的代码

解释说明。

Android.mk 编写:

#每个Android.mk文件必须以定义LOCAL_PATH为开始。它用于在开发tree中查找源文件

LOCAL_PATH := $(call my-dir)

常用的一些 函数

  1. $(call my-dir) #获取当前文件夹的路径。
  2. $(call all-java-files-under, <src>) #获取指定目录下的所有java文件。
  3. $(call all-c-files-under, <src>) #获取指定目录下的所有c文件。
  4. $(call all-Iaidl-files-under, <src>) #获取指定目录下的所有AIDL文件。
  5. $(call all-makefiles-under, <folder>) #获取指定目录下的所有Make文件。
  6. $(call intermediates-dir-for, <class>, <app_name>, <host or target>, <common?>) #获取Build输入的目标文件夹路径。
  7. $(call first-makefiles-under,$(LOCAL_PATH)) #在LOCAL_PATH的所有子目录中查找.mk文件,不包括当前目录

#CLEAR_VARS 变量由Build System提供。并指向一个指定的GNU Makefile,由它负责清理很多 LOCAL_xxx.例如:LOCAL_MODULE, LOCAL_SRC_FILES, LOCAL_STATIC_LIBRARIES, 但不清理LOCAL_PATH,这个清理动作是必须的,因为所有的编译控制文件由同一个GNU Make解析和执行,其变量是全局的。所以清理后才能避免相互影响。

include $(CLEAR_VARS)

#LOCAL_MODULE_TAGS :=  user eng tests optional

user: 指该模块只在user版本下才编译

eng: 指该模块只在eng版本下才编译

tests: 指该模块只在tests版本下才编译

optional:指该模块在所有版本下都编译

  1. LOCAL_MODULE_TAGS := optional

#LOCAL_SRC_FILES变量必须包含将要打包如模块的C/C++ , java源码。

LOCAL_SRC_FILES := $(call all-java-files-under, app/src/main/java)

#编译AndriodManifest.xml

LOCAL_MANIFEST_FILE := app/src/main/AndroidManifest.xml


# 编译的系统资源文件

LOCAL_RESOURCE_DIR := $(LOCAL_PATH)/app/src/main/res


# 编译asset资源文件,必须要单独写出来,不然在工程中无法访问到asset路径下的资源文件

LOCAL_ASSET_DIR := $(LOCAL_PATH)/app/src/main/assets


#指定APP应用名称

LOCAL_PACKAGE_NAME := MyTestApp

#在指定分区中安装此模块

     一般有  system    system_ext     product     vendor     odm       分区     

  1. #模块编译输出分区
  2. #system :主要包含 Android 框架, google 官方实现
  3. #Android.mk 默认就是输出到 system 分区,不用指定
  4. #Android.bp 默认就是输出到 system 分区,不用指定
  5. #system_ext: android11 新划分区
  6. #vendor :SoC芯片商分区(系统级核心厂商,如高通), 为他们提供一些核心功能和服务,由 soc 实现
  7. #Android.mk LOCAL_VENDOR_MODULE := true
  8. #Android.bp vendor: true
  9. #odm :设备制造商分区(如华为、小米),为他们的传感器或外围设备提供一些核心功能和服务
  10. #Android.mk LOCAL_ODM_MODULE := true
  11. #Android.bp device_specific: true
  12. #product :产品机型分区
  13. #Android.mk LOCAL_PRODUCT_MODULE := true
  14. #Android.bp product_specific
  1. #安装到product分区
  2. LOCAL_PRODUCT_MODULE := true

#LOCAL_PRIVILEGED_MODULE := true对于Android系统应用LOCAL_PRIVILEGED_MODULE 决定了app在编译后,放在ROM分区中的 app  或者 priv-app位置

  1. LOCAL_PRIVILEGED_MODULE := true
  2. #如果配置为false 则编译到设备中的路径为 (system or product or vendor/app)
  3. #如果配置为true 则编译到设备中的路径为 (system or product or vendor/priv-app)

#使用sdk的hide的api來编译, Android P 之后,Android.mk 必须定义 LOCAL_SDK_VERSION 和 LOCAL_PRIVATE_PLATFORM_APIS 变量中的一个, 这两种是或的关系,只需要定义一个

  1. #如果需要使用系统隐藏 API编译 则需要定义:
  2. LOCAL_PRIVATE_PLATFORM_APIS := true
  3. #如果不需要使用系统隐藏API编译,则需要定义:
  4. LOCAL_SDK_VERSION := current
  5. #注意,两个是 if --else 的定义原则,根据实际情况选择

LOCAL_MODULE_CLASS   指定模块的类型,可不用定义,可以选择不定义

  1. 有如下几种:
  2. 编译 apk 文件 ,最后生成的文件路径一般放在 /system/app目录下
  3. LOCAL_MODULE_CLASS := APPS
  4. 编译 jar 包
  5. LOCAL_MODULE_CLASS := JAVA_LIBRAYIES
  6. 定义动态库文件 最后生成的文件路径放置在/sysem/lib目录下
  7. LOCAL_MODULE_CLASS := SHARED_LIBRAYIES
  8. 编译可执行文件 最后生成的文件路径放置放在system/bin下
  9. LOCAL_MODULE_CLASS := EXECUTABLES
  10. 编译配置文件 最后生成的文件路径放置在 /system/etc/目录下
  11. LOCAL_MODULE_CLASS := ETC

 #aapt 是编译和打包资源的工具。而aapt2是在aapt上做了优化

LOCAL_USE_AAPT2 := true   

#指定依赖的共享java类库,这个是编译时依赖,最终不会打包到apk中

LOCAL_JAVA_LIBRARIES := javax.obex telephony-common services.net    

#指定依赖的静态java类库,最终会打包到apk里面。

 使用多个引用的时候,写法如下:

  1. LOCAL_STATIC_JAVA_LIBRARIES := \
  2. com.android.vcard \
  3. bluetooth.cc\
  4. services.net \
  5. libprotobuf-java-lite \

#指定依赖的模块。指定模块运行所依赖的模块(模块安装时将会同步安装它所依赖的模块)

LOCAL_REQUIRED_MODULES := libjni_pinyin_test

#混淆配置,默认为full obfuscation,全代码混淆,disabled不开启

  1. #不需要使用代码混淆的工具进行代码混淆
  2. LOCAL_PROGUARD_ENABLED := disabled
  3. #默认为: 使用将该工程代码全部混淆
  4. LOCAL_PROGUARD_ENABLE := full

# LOCAL_CERTIFICATE 用于设置不同的签名方式

        LOCAL_CERTIFICATE := PRESIGNED,打包apk时,沿用apk中原来的签名

用于设置不同的签名方式build/target/product/security目录中有四组默认签名供Android.mk在编译APK使用:

        1、testkey:普通APK,默认情况下使用。

        2、platform:该APK完成一些系统的核心功能。经过对系统中存在的文件夹的访问测试,这种方式编译出来的APK所在进程的UID为system。

        3、shared:该APK需要和home/contacts进程共享数据。

        4、media:该APK是media/download系统中的一环。

应用程序的Android.mk中有一个LOCAL_CERTIFICATE字段,由它指定用哪个key签名,未指定的默认用testkey.

  1. LOCAL_CERTIFICATE : #使用平台文件签名
  2. 1)platform签名:
  3. AndroidManifest.xml的manifest节点中添加 android:sharedUserId=”android.uid.system”,
  4. Android.mk中增加  LOCAL_CERTIFICATE := platform
  5. 2)shared签名:
  6. AndroidManifest.xml的manifest节点中增加android:sharedUserId=”android.uid.shared”,
  7. Android.mk中增加LOCAL_CERTIFICATE := shared
  8. 3)media签名:
  9. AndroidManifest.xml的manifest节点中增加 android:sharedUserId=”android.media”,
  10. Android.mk中增加 LOCAL_CERTIFICATE := media

Settings.apk就是platform级别的签名,系统级应用都应该使用这个签名

LOCAL_CERTIFICATE := platform

#声明要调用android包,会打包到apk中,调用多个时,写法如下:

  1. LOCAL_STATIC_ANDROID_LIBRARIES := \
  2. androidx.appcompat_appcompat \
  3. com.google.android.material_material \
  4. androidx-constraintlayout_constraintlayout \
  5. lottie-2.8.0 //lottie 第三方aar

#aapt是Android的打包工具,

1、–auto-add-overlay意思是当不同文件夹有相同的资源id时,只将第一个资源合并打包进来,
2、–extra-packages android.support.v7.appcompat,意思是将extra-package包后边的文件包引入到当前代码框架中,就像给当前代码导入一个jar包,但是这里不限jar、library等等,这样引入编译之后就能合并在一起使用

  1. LOCAL_AAPT_FLAGS := \
  2. --auto-add-overlay \
  3. --extra-packages com.airbnb.lottie //lottie 第三方aar的包名

#编译APK

include $(BUILD_PACKAGE)


延伸定义

BUILD_HOST_STATIC_LIBRARY 编译静态库(适用与主机)
BUILD_HOST_SHARED_LIBRARY 编译动态库(适用与主机)
BUILD_HOST_EXECUTABLE 编译可执行程序(适用与主机)
BUILD_HOST_PREBUILT 预编译(适用与主机)
BUILD_HOST_JAVA_LIBRARY 编译java包(适用与主机)
BUILD_JAVA_LIBRARY 编译java包
BUILD_STATIC_JAVA_LIBRARY 编译java静态包


BUILD_STATIC_LIBRARY 编译静态库
BUILD_SHARED_LIBRARY 编译动态库

  1. 官方解释是这样:
  2. 1、 LOCAL_STATIC_JAVA_LIBRARIES-静态库是在连接阶段直接拷贝到代码 中使用的,BUILD_STATIC_JAVA_LIBRARY指定生成静态库。编译出来的静态库(这里指jar包)里每个java文件对应的class文件都单独存在,可以直接导入Eclipse等IDE使用,
  3. 2、LOCAL_JAVA_LIBRARIES -而共享库是由加载器加载到内存,在运行时使用的。而编译出来的共享库(jar包),内部是Android字节码Dex格式的文件,一般无法导入Eclipse等IDE使用。Android.mk中由BUILD _ JAVA _ LIBRARY指定生成共享库
  4. 静态库和动态库本质上没有区别,可能就是说静态库是编译时期就把jar包中的代码整合到项目中了,动态库就是说在程序运行期,把jar包加载到内存中,每次使用再继续重新加载到内存


BUILD_EXECUTABLE 编译可执行程序
BUILD_PACKAGE 编译apk  ---- 需要有源码文件的app模块
BUILD_PREBUILT 预编译(针对单个预编译文件)  
BUILD_MULTI_PREBUILT 预编译(针对多个预编译文件)
 

LOCAL_JAVA_LIBRARIES 和 LOCAL_STATIC_JAVA_LIBRARIES 区别

LOCAL_STATIC_JAVA_LIBRARIES
LOCAL_STATIC_JAVA_LIBRARIES表示把引用的外部Java库直接编译打包到本模块中,在runtime时可以直接从本模块中找到相关的东西。

LOCAL_JAVA_LIBRARIES
LOCAL_JAVA_LIBRARIES表示引用的外部Java库在编译时可以找到相关的东西,但并不打包到本模块,在runtime时需要从别的地方查找,这个别的地方就是在编译时将引用的外部Java库的模块名添加到PRODUCT_BOOT_JARS,例如Android原生的framework.jar就添了,PRODUCT_BOOT_JARS+=framework,否则在runtime时ClassLoader找不到需要的class。

综上,Android中所谓的Java动态库、静态库不同于C、C++的动态库、静态库,编译为静态库时表示编译为原生的Java class压缩包,编译为动态库时表示编译为用于Android Dalvik的dex压缩包,静态引用时把外部库直接打包过来,动态引用时需要将外部库加到PRODUCT_BOOT_JARS。

#Android.mk 文件结尾,make文件最后一句话

include $(call all-makefiles-under,$(LOCAL_PATH))

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Cpp五条/article/detail/562469
推荐阅读
相关标签
  

闽ICP备14008679号