AndroidManifest合并原理

2017-03-22  本文已影响0人  jianhuih

3.11 AndroidManifest合并原理

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

3.11.1 合并冲突规则(merge conflict rules)

合并冲突,是指多个Manifest文件中含有同一属性但值不同时,默认合并规则解决不了从而导致的冲突。当冲突发生时,高优先级的Manifest属性值会覆盖低优先级属性值。这个优先级规则由高到低依次是:

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

默认合并冲突规则如下:

当然还存在例外情况:

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

合并冲突标记,是android tools namespace中的一个属性,用来解决默认冲突规则解决不了的冲突。主要包含以下几个:

一般节点层面默认使用merge,属性层面默认使用strict。下面看几个例子:

  1. 使用replace标记解决android:icon和android:label属性冲突
<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>
  1. 以下代码块中,src manifest会覆盖library的<uses-sdk>。(默认情况下是不允许低优先级的minSdkVersion大于高优先级的,否则会报错。)
//src manifest
<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"/>
   ...
</manifest>
//Library manifest
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.lib1">
    ...
    <uses-sdk android:minSdkVersion="4" />
    ...
</manifest>
  1. 以下代码块表示,移除library1中的permissionOne权限,而其他模块下该权限不受影响。
<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.11.3 向AndroidManifest文件注入build变量值

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

  1. 注入applicationId
<activity android:name=".Main">
    <intent-filter>
        <action android:name="${applicationId}.foo"></action>
    </intent-filter>
</activity>

Gradle build file:

android {
    compileSdkVersion 22
    buildToolsVersion "22.0.1"

    productFlavors {
        flavor1 {
            applicationId = "com.mycompany.myapplication.productFlavor1"
        }
    }
    ...
}

注入之后的manifest为:

<action android:name="com.mycompany.myapplication.productFlavor1.foo"/>
  1. 注入其他属性
    Gradle build file:
android {
    defaultConfig {
        manifestPlaceholders = [ activityLabel:"defaultName"]
    }
    productFlavors {
        free {
        }
        pro {
            manifestPlaceholders = [ activityLabel:"proName" ]
        }
    }
    ...
}

Placeholder in the manifest file:

<activity android:name=".MainActivity" android:label="${activityLabel}" />
上一篇 下一篇

猜你喜欢

热点阅读