为了账号安全,请及时绑定邮箱和手机立即绑定

AndroidManifest合并原理

标签:
Android

Android Studio工程通常包含多个AndroidManifest文件,最终构建成APK时,会合并成一个AndroidManifest文件。但是可能很多人应该都不知道是怎么合并的,本文将为大家揭开神秘面纱。

1. 合并冲突规则(merge conflict rules)

合并冲突,是指多个Manifest文件中含有同一属性但值不同时,默认合并规则解决不了从而导致的冲突。

当冲突发生时,高优先级的Manifest属性值会覆盖低优先级属性值。这个优先级规则由高到低依次是:

?

1

buildType下的Manifest设置->productFlavor下的Manifest设置->主工程src/main->dependency&library

默认合并冲突规则如下:

5bc31cdb0001ec0a15240664.jpg

当然还存在例外情况:

1.     uses-feature android:required和uses-library android:required默认值都是true,根据OR规则合并

2.     如果不指定uses-sdk,默认的minSdkVersion和targetSdkVersion值为1,当发生冲突时将使用高优先级的值。若不指定targetSdkVersion,其值等于targetSdkVersion

3.     当library工程的minSdkVersion比主工程src/main中的minSdkVersion低时会产生冲突,此时需要添加overLibrary标记解决冲突

4.     当library工程的targetSdkVersion比主工程src/main中的大时,合并过程会增加一些权限保证library工程能正常运行

5.     每个Manifest文件只和其子Manifest文件的属性合并

6.     <intent-filter>的合并规则是叠加而不是覆盖

2. 合并冲突标记和选择器(merge conflict marker&selector)

合并冲突标记,是android tools namespace中的一个属性,用来解决默认冲突规则解决不了的冲突。

主要包含以下几个:

1.     merge

默认合并操作。

2.     replace

高优先级替换低优先级Manifest文件中的属性

3.     strict

属性相同而值不同时会报错,除非通过冲突规则resolved

4.     merge-only

仅合并低优先级的属性

5.     remove

移除指定的低优先级的属性

6.     remove-All

移除相同节点类型下所有低优先级的属性

注:节点层面默认使用merge,属性层面默认使用strict

下面看几个例子:

(1)使用replace标记解决androidk:icon和android:label属性冲突

[代码]xml代码:

?

1

2

3

4

<manifest xmlns:android="http://schemas.android.com/apk/res/android"   package="com.android.tests.flavorlib.app" xmlns:tools="http://schemas.android.com/tools">

 

   <application android:icon="@drawable/icon"   android:label="@string/app_name" tools:replace="icon,   label">

       ...</application></manifest>

(2)以下代码块中,src/main会覆盖library的<uses-sdk>。(默认情况下是不允许低优先级的minSdkVersion大于高优先级的,否则会报错。)

src/main/ Manifest:

[代码]xml代码:

?

1

2

3

<manifest xmlns:android="http://schemas.android.com/apk/res/android"   package="com.android.example.app" xmlns:tools="http://schemas.android.com/tools">

   ...   <uses-sdk android:targetsdkversion="22"   android:minsdkversion="2" tools:overridelibrary="com.example.lib1,   com.example.lib2">

   ...</uses-sdk></manifest>

Library manifest:

[代码]xml代码:

?

1

2

3

<manifest xmlns:android="http://schemas.android.com/apk/res/android"   package="com.example.lib1">

    ...      <uses-sdk android:minsdkversion="4">

    ...</uses-sdk></manifest>

(3)以下代码块表示,移除library1中的permissionOne权限,而其他模块下该权限不受影响。

[代码]xml代码:

?

1

2

3

<manifest xmlns:android="http://schemas.android.com/apk/res/android"   package="com.android.example.app" xmlns:tools="http://schemas.android.com/tools">

   ...   <permission android:name="permissionOne"   tools:node="remove" tools:selector="com.example.lib1">

       ...</permission></manifest>

3. 向AndroidManifest文件注入build变量值

注入build变量值通常需要使用manifestPlaceholders,applicationId属性除外。另外支持部分注入,如android:authority="com.acme.${localApplicationId}.foo".

直接看例子吧:

(1)注入applicationId

Manifest entry:

[代码]xml代码:

?

1

2

3

4

<activity android:name=".Main">

    <intent-filter>

        <action   android:name="${applicationId}.foo"></action>

    </intent-filter></activity>

Gradle build file:

[代码]xml代码:

?

1

2

3

4

5

6

7

8

9

android {

    compileSdkVersion 22

    buildToolsVersion   "22.0.1"

 

    productFlavors {

        flavor1   {

            applicationId   = "com.mycompany.myapplication.productFlavor1"

        }

    }

Merged manifest value:

[代码]xml代码:

?

1

<action android:name="com.mycompany.myapplication.productFlavor1.foo"   data-filtered="filtered"></action>

(2)注入其他属性

Gradle build file:

[代码]xml代码:

?

01

02

03

04

05

06

07

08

09

10

11

android {

    defaultConfig {

        manifestPlaceholders   = [ activityLabel:"defaultName"]

    }

    productFlavors {

        free {

        }

        pro {

            manifestPlaceholders   = [ activityLabel:"proName" ]

        }

    }

Placeholder in the manifest file:

[代码]xml代码:

?

1

<activity android:name=".MainActivity" android:label="${activityLabel}"   data-filtered="filtered"></activity>

4. 基于product favors group的合并(貌似没咋看懂,有知道的朋友可以告诉我~)

5bc31cc40001ef2f15240336.jpg

AndroidManifest合并顺序为:

prod-paid -> API-22-> density-mdpi -> ABI-x86

5. 关于隐藏的权限

当导入的library工程里隐含一些默认的权限时,合并过程会把这些默认权限加进来,以保证不同版本的sdk能够正常运行。如:

5bc31cb80001e43c15160546.jpg

6. 关于merge错误处理

程序构建过程中,合并过程生成日志位于build/outputs/logs的manifest-merger-<productFlavor>-report.txt

Higher priority manifest declaration:

[代码]xml代码:

?

1

<activity android:name="com.foo.bar.ActivityOne"   android:screenorientation="portrait" android:theme="@theme1"   data-filtered="filtered"></activity>

Lower priority manifest declaration:

[代码]xml代码:

?

1

<activity android:name="com.foo.bar.ActivityOne"   android:screenorientation="landscape" data-filtered="filtered"></activity>

Error log:

?

1

/project/app/src/main/AndroidManifest.xml:3:9 Error:Attribute   activity@screenOrientation value=(portrait) from AndroidManifest.xml:3:9is   also present at flavorlib:lib1:unspecified:3:18 value=(landscape)Suggestion:   add 'tools:replace="icon"' to element at AndroidManifest.xml:1:5 to   override

原文链接:http://www.apkbus.com/blog-705730-60674.html

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消