3.11 AndroidManifest合并原理
Android Studio工程通常包含多個AndroidManifest文件蜂筹,最終構(gòu)建成APK時力细,會合并成一個AndroidManifest文件纪铺。但是可能很多人應該都不知道是怎么合并的,本文將為大家揭開神秘面紗料按。
3.11.1 合并沖突規(guī)則(merge conflict rules)
合并沖突篡殷,是指多個Manifest文件中含有同一屬性但值不同時,默認合并規(guī)則解決不了從而導致的沖突藻治。當沖突發(fā)生時碘勉,高優(yōu)先級的Manifest屬性值會覆蓋低優(yōu)先級屬性值。這個優(yōu)先級規(guī)則由高到低依次是:
buildType下的Manifest設置->productFlavor下的Manifest設置->主工程src/main->dependency&library
默認合并沖突規(guī)則如下:
當然還存在例外情況:
- uses-feature android:required和uses-library android:required默認值都是true桩卵,根據(jù)OR規(guī)則合并
- 如果不指定uses-sdk验靡,默認的minSdkVersion和targetSdkVersion值為1,當發(fā)生沖突時將使用高優(yōu)先級的值雏节。若不指定targetSdkVersion胜嗓,其值等于targetSdkVersion
- 當library工程的minSdkVersion比主工程src/main中的minSdkVersion低時會產(chǎn)生沖突,此時需要添加overLibrary標記解決沖突
- 當library工程的targetSdkVersion比主工程src/main中的大時钩乍,合并過程會增加一些權限保證library工程能正常運行
- 每個Manifest文件只和其子Manifest文件的屬性合并
- <intent-filter>的合并規(guī)則是疊加而不是覆蓋
3.11.2 合并沖突標記和選擇器(merge conflict marker&selector)
合并沖突標記辞州,是android tools namespace中的一個屬性,用來解決默認沖突規(guī)則解決不了的沖突寥粹。主要包含以下幾個:
- merge:默認合并操作变过。
- replace:高優(yōu)先級替換低優(yōu)先級Manifest文件中的屬性
- strict:屬性相同而值不同時會報錯埃元,除非通過沖突規(guī)則解決了
- merge-only:僅合并低優(yōu)先級的屬性
- remove:移除指定的低優(yōu)先級的屬性
- remove-All:移除相同節(jié)點類型下所有低優(yōu)先級的屬性
一般節(jié)點層面默認使用merge,屬性層面默認使用strict媚狰。下面看幾個例子:
- 使用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>
- 以下代碼塊中岛杀,src manifest會覆蓋library的<uses-sdk>。(默認情況下是不允許低優(yōu)先級的minSdkVersion大于高優(yōu)先級的崭孤,否則會報錯类嗤。)
//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>
- 以下代碼塊表示,移除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"。仍然是看幾個例子:
- 注入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"/>
- 注入其他屬性
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}" />