मेनिफ़ेस्ट फ़ाइलें प्रबंधित करें

इस पेज पर, मेनिफ़ेस्ट मर्ज करने के तरीके के बारे में बताया गया है. साथ ही, मर्ज से जुड़ी समस्याओं को हल करने के लिए, मर्ज की प्राथमिकताएं लागू करने का तरीका भी बताया गया है. ऐप्लिकेशन मेनिफ़ेस्ट फ़ाइल के बारे में जानने के लिए, ऐप्लिकेशन मेनिफ़ेस्ट की खास जानकारी लेख पढ़ें.

एक से ज़्यादा मेनिफ़ेस्ट फ़ाइलों को मर्ज करना

आपकी APK या Android ऐप्लिकेशन बंडल फ़ाइल में सिर्फ़ एक AndroidManifest.xml फ़ाइल हो सकती है. हालांकि, आपके Android Studio प्रोजेक्ट में मुख्य सोर्स सेट, बिल्ड वैरिएंट, और इंपोर्ट की गई लाइब्रेरी से मिली कई मेनिफ़ेस्ट फ़ाइलें हो सकती हैं. आपके ऐप्लिकेशन को बिल्ड करते समय, Gradle बिल्ड सभी मेनिफ़ेस्ट फ़ाइलों को एक मेनिफ़ेस्ट फ़ाइल में मर्ज कर देता है. इस फ़ाइल को आपके ऐप्लिकेशन में पैकेज किया जाता है.

मेनिफ़ेस्ट मर्ज करने वाला टूल, हर फ़ाइल के सभी एक्सएमएल एलिमेंट को मर्ज करता है. इसके लिए, वह एक्सएमएल एलिमेंट को मर्ज करने के लिए इस्तेमाल किए जाने वाले तरीकों का पालन करता है. साथ ही, उन प्राथमिकताओं का पालन करता है जिन्हें आपने खास एक्सएमएल एट्रिब्यूट की मदद से तय किया है.

सलाह: मर्ज किए गए मेनिफ़ेस्ट के नतीजों की झलक देखने और विरोध की गड़बड़ियों का पता लगाने के लिए, मर्ज किए गए मेनिफ़ेस्ट के व्यू का इस्तेमाल करें. इस बारे में अगले सेक्शन में बताया गया है.

प्राथमिकताएं मर्ज करना

मर्ज करने वाला टूल, सभी मेनिफ़ेस्ट फ़ाइलों को एक फ़ाइल में क्रम से जोड़ता है. यह फ़ाइल, हर मेनिफ़ेस्ट फ़ाइल की प्राथमिकता के आधार पर बनाई जाती है. उदाहरण के लिए, अगर आपके पास तीन मेनिफ़ेस्ट फ़ाइलें हैं, तो सबसे कम प्राथमिकता वाले मेनिफ़ेस्ट को सबसे ज़्यादा प्राथमिकता वाले मेनिफ़ेस्ट में मर्ज किया जाता है. इसके बाद, सबसे ज़्यादा प्राथमिकता वाले मेनिफ़ेस्ट को सबसे कम प्राथमिकता वाले मेनिफ़ेस्ट में मर्ज किया जाता है, जैसा कि पहले चित्र में दिखाया गया है.

पहली इमेज. तीन मेनिफ़ेस्ट फ़ाइलों को मर्ज करने की प्रोसेस, सबसे कम प्राथमिकता से सबसे ज़्यादा प्राथमिकता में.

मेनिफ़ेस्ट फ़ाइलें तीन तरह की होती हैं, जिन्हें एक-दूसरे में मर्ज किया जा सकता है. साथ ही, मर्ज करने की प्राथमिकताएं इस तरह से होती हैं (सबसे ज़्यादा प्राथमिकता पहले):

  1. आपके बिल्ड वैरिएंट के लिए मेनिफ़ेस्ट फ़ाइल

    अगर आपके पास अपने वैरिएंट के लिए एक से ज़्यादा सोर्स सेट हैं, तो उनके मेनिफ़ेस्ट की प्राथमिकताएं इस तरह से हैं:

    • बिल्ड का वैरिएंट मेनिफ़ेस्ट (जैसे, src/demoDebug/)
    • बिल्ड टाइप मेनिफ़ेस्ट (जैसे, src/debug/)
    • प्रॉडक्ट फ़्लेवर मेनिफ़ेस्ट (जैसे, src/demo/)

      अगर फ़्लेवर डाइमेंशन का इस्तेमाल किया जा रहा है, तो मेनिफ़ेस्ट की प्राथमिकताएं, flavorDimensions प्रॉपर्टी में हर डाइमेंशन के क्रम से मेल खाती हैं. पहला डाइमेंशन सबसे ज़्यादा प्राथमिकता वाला होता है.

  2. ऐप्लिकेशन मॉड्यूल के लिए मुख्य मेनिफ़ेस्ट फ़ाइल
  3. शामिल की गई लाइब्रेरी की मेनिफ़ेस्ट फ़ाइल

    अगर आपके पास एक से ज़्यादा लाइब्रेरी हैं, तो उनके मेनिफ़ेस्ट की प्राथमिकताएं, आपके Gradledependencies ब्लॉक में दिखने के क्रम से मैच होती हैं.

उदाहरण के लिए, लाइब्रेरी मेनिफ़ेस्ट को मुख्य मेनिफ़ेस्ट में मर्ज किया जाता है. इसके बाद, मुख्य मेनिफ़ेस्ट को बिल्ड वैरिएंट मेनिफ़ेस्ट में मर्ज किया जाता है. ध्यान दें कि ये सभी सोर्स सेट के लिए, मर्ज करने की एक जैसी प्राथमिकताएं हैं, जैसा कि सोर्स सेट की मदद से बनाएं में बताया गया है.

अहम जानकारी: build.gradle फ़ाइल से मिले बिल्ड कॉन्फ़िगरेशन, मर्ज की गई मेनिफ़ेस्ट फ़ाइल में मौजूद मिलते-जुलते एट्रिब्यूट को बदल देते हैं. उदाहरण के लिए, build.gradle या build.gradle.kts फ़ाइल का minSdk, <uses-sdk> मैनिफ़ेस्ट एलिमेंट में मौजूद मैच करने वाले एट्रिब्यूट को बदल देता है. भ्रम से बचने के लिए, <uses-sdk> एलिमेंट को हटाएं और इन प्रॉपर्टी को सिर्फ़ build.gradle फ़ाइल में तय करें. ज़्यादा जानकारी के लिए, अपना बिल्ड कॉन्फ़िगर करना लेख पढ़ें.

मर्ज करने से जुड़े ह्यूरिस्टिक्स

मर्ज करने वाला टूल, एक मेनिफ़ेस्ट के हर एक्सएमएल एलिमेंट को दूसरे मेनिफ़ेस्ट के मिलते-जुलते एलिमेंट से लॉजिक के हिसाब से मैच कर सकता है. मैच करने के तरीके के बारे में जानकारी पाने के लिए, पिछले सेक्शन में मर्ज की प्राथमिकताएं देखें.

अगर कम प्राथमिकता वाले मेनिफ़ेस्ट का कोई एलिमेंट, ज़्यादा प्राथमिकता वाले मेनिफ़ेस्ट के किसी भी एलिमेंट से मेल नहीं खाता है, तो उसे मर्ज किए गए मेनिफ़ेस्ट में जोड़ दिया जाता है. हालांकि, अगर कोई मैच करने वाला एलिमेंट मौजूद है, तो मर्ज करने वाला टूल, हर एलिमेंट के सभी एट्रिब्यूट को एक ही एलिमेंट में जोड़ने की कोशिश करता है. अगर टूल को पता चलता है कि दोनों मेनिफ़ेस्ट में एक ही एट्रिब्यूट की अलग-अलग वैल्यू मौजूद हैं, तो मर्ज करने में समस्या आती है.

टेबल 1 में, उन संभावित नतीजों को दिखाया गया है जो मर्ज टूल के सभी एट्रिब्यूट को एक ही एलिमेंट में जोड़ने की कोशिश करने पर मिल सकते हैं.

टेबल 1. एट्रिब्यूट वैल्यू के लिए मर्ज करने का डिफ़ॉल्ट तरीका

ज़्यादा प्राथमिकता वाला एट्रिब्यूट कम प्राथमिकता वाला एट्रिब्यूट एट्रिब्यूट का मर्ज किया गया नतीजा
काम की नहीं है काम की नहीं है कोई वैल्यू नहीं (डिफ़ॉल्ट वैल्यू का इस्तेमाल करें)
वैल्यू B वैल्यू B
वैल्यू A काम की नहीं है वैल्यू A
वैल्यू A वैल्यू A
वैल्यू B कॉन्फ़िगरेशन में गड़बड़ी—आपको एक मर्ज नियम मार्कर जोड़ना होगा.

हालांकि, कुछ स्थितियों में मर्ज करने के टूल का व्यवहार अलग होता है, ताकि मर्ज करने से जुड़ी समस्याओं से बचा जा सके:

  • <manifest> एलिमेंट में मौजूद एट्रिब्यूट को कभी भी एक साथ मर्ज नहीं किया जाता. सिर्फ़ सबसे ज़्यादा प्राथमिकता वाले मेनिफ़ेस्ट के एट्रिब्यूट का इस्तेमाल किया जाता है.
  • <uses-feature> और <uses-library> एलिमेंट में मौजूद android:required एट्रिब्यूट, OR मर्ज का इस्तेमाल करता है. अगर कोई विरोध होता है, तो "true" लागू होता है और किसी एक मेनिफ़ेस्ट में ज़रूरी सुविधा या लाइब्रेरी हमेशा शामिल होती है.
  • <uses-sdk> एलिमेंट में मौजूद एट्रिब्यूट, हमेशा ज़्यादा प्राथमिकता वाले मेनिफ़ेस्ट की वैल्यू का इस्तेमाल करते हैं. हालांकि, इन स्थितियों में ऐसा नहीं होता:
    • जब कम प्राथमिकता वाले मेनिफ़ेस्ट की minSdk वैल्यू, ज़्यादा होती है, तो तब तक गड़बड़ी होती है, जब तक कि आपने overrideLibrary मर्ज करने का नियम लागू नहीं किया है.
    • जब कम प्राथमिकता वाले मेनिफ़ेस्ट में targetSdkVersion वैल्यू कम होती है, तो मर्ज करने वाला टूल, ज़्यादा प्राथमिकता वाले मेनिफ़ेस्ट की वैल्यू का इस्तेमाल करता है. साथ ही, वह ऐसी सभी सिस्टम अनुमतियां भी जोड़ता है जो यह पक्का करने के लिए ज़रूरी हैं कि इंपोर्ट की गई लाइब्रेरी सही तरीके से काम करती रहे. ऐसा उन मामलों में किया जाता है जिनमें Android के नए वर्शन में, अनुमतियों से जुड़ी पाबंदियां बढ़ाई गई हैं. इस व्यवहार के बारे में ज़्यादा जानकारी के लिए, सिस्टम की अनुमतियों वाला सेक्शन देखें.
  • <intent-filter> एलिमेंट, एक मेनिफ़ेस्ट से दूसरे मेनिफ़ेस्ट के साथ कभी मैच नहीं होता. हर एलिमेंट को यूनीक माना जाता है और उसे मर्ज किए गए मेनिफ़ेस्ट में, सामान्य पैरंट एलिमेंट में जोड़ा जाता है.

एट्रिब्यूट के बीच होने वाले अन्य सभी विरोधों के लिए, आपको गड़बड़ी का मैसेज मिलता है. साथ ही, आपको ज़्यादा प्राथमिकता वाली मेनिफ़ेस्ट फ़ाइल में कोई खास एट्रिब्यूट जोड़कर, मर्ज टूल को इसे ठीक करने का तरीका बताना होगा. मर्ज नियम मार्कर के बारे में जानने के लिए, यहां दिया गया सेक्शन देखें.

एट्रिब्यूट की डिफ़ॉल्ट वैल्यू पर निर्भर न रहें. सभी यूनीक एट्रिब्यूट को एक ही एलिमेंट में जोड़ दिया जाता है. इसलिए, अगर ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट, किसी एट्रिब्यूट की डिफ़ॉल्ट वैल्यू पर निर्भर करता है, तो इससे अनचाहे नतीजे मिल सकते हैं. उदाहरण के लिए, अगर प्राथमिकता में सबसे ऊपर वाले मेनिफ़ेस्ट में android:launchMode एट्रिब्यूट की जानकारी नहीं दी गई है, तो वह "standard" की डिफ़ॉल्ट वैल्यू का इस्तेमाल करता है. हालांकि, अगर प्राथमिकता में सबसे नीचे वाले मेनिफ़ेस्ट में इस एट्रिब्यूट की जानकारी किसी दूसरी वैल्यू के साथ दी गई है, तो वह वैल्यू मर्ज किए गए मेनिफ़ेस्ट पर लागू हो जाती है और डिफ़ॉल्ट वैल्यू को बदल देती है. आपको हर एट्रिब्यूट को साफ़ तौर पर वैसा ही तय करना चाहिए जैसा आपको चाहिए. हर एट्रिब्यूट की डिफ़ॉल्ट वैल्यू, मेनिफ़ेस्ट रेफ़रंस में दी गई होती हैं.

नियम मार्कर मर्ज करना

मर्ज नियम मार्कर एक एक्सएमएल एट्रिब्यूट है. इसका इस्तेमाल, मर्ज से जुड़ी समस्याओं को हल करने या अनचाहे एलिमेंट और एट्रिब्यूट को हटाने के तरीके के बारे में अपनी प्राथमिकता बताने के लिए किया जा सकता है. मार्कर को पूरे एलिमेंट या एलिमेंट के कुछ खास एट्रिब्यूट पर लागू किया जा सकता है.

दो मेनिफ़ेस्ट फ़ाइलों को मर्ज करते समय, मर्ज करने वाला टूल, ज़्यादा प्राथमिकता वाली मेनिफ़ेस्ट फ़ाइल में इन मार्कर को खोजता है.

सभी मार्कर, Android tools नेमस्पेस से जुड़े होते हैं. इसलिए, आपको सबसे पहले <manifest> एलिमेंट में इस नेमस्पेस का एलान करना होगा, जैसा कि यहां दिखाया गया है:

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

नोड मार्कर

पूरे एक्सएमएल एलिमेंट (किसी दिए गए मेनिफ़ेस्ट एलिमेंट के सभी एट्रिब्यूट और उसके सभी चाइल्ड टैग पर) पर मर्ज करने का नियम लागू करने के लिए, इन एट्रिब्यूट का इस्तेमाल करें:

tools:node="merge"
विरोध को हल करने के लिए, एलिमेंट को मर्ज करने के तरीके का इस्तेमाल करके, इस टैग और सभी नेस्ट किए गए एलिमेंट के सभी एट्रिब्यूट को मर्ज करें. ऐसा तब करें, जब कोई विरोध न हो. यह एलिमेंट के लिए डिफ़ॉल्ट व्यवहार है.

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:windowSoftInputMode="stateUnchanged">
    <intent-filter>
        <action android:name="android.intent.action.SEND" />
        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>
</activity>

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:screenOrientation="portrait"
    tools:node="merge">
</activity>

मर्ज किए गए मेनिफ़ेस्ट का नतीजा:

<activity android:name="com.example.ActivityOne"
    android:screenOrientation="portrait"
    android:windowSoftInputMode="stateUnchanged">
    <intent-filter>
        <action android:name="android.intent.action.SEND" />
        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>
</activity>
tools:node="merge-only-attributes"
सिर्फ़ इस टैग में एट्रिब्यूट मर्ज करें. नेस्ट किए गए एलिमेंट को मर्ज न करें.

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:windowSoftInputMode="stateUnchanged">
    <intent-filter>
        <action android:name="android.intent.action.SEND" />
        <data android:type="image/*" />
        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>
</activity>

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:screenOrientation="portrait"
    tools:node="merge-only-attributes">
</activity>

मर्ज किए गए मेनिफ़ेस्ट का नतीजा:

<activity android:name="com.example.ActivityOne"
    android:screenOrientation="portrait"
    android:windowSoftInputMode="stateUnchanged">
</activity>
tools:node="remove"
मर्ज किए गए मेनिफ़ेस्ट से इस एलिमेंट को हटाएं. इसका इस्तेमाल तब किया जाता है, जब आपको मर्ज किए गए मेनिफ़ेस्ट में ऐसा एलिमेंट मिलता है जिसकी आपको ज़रूरत नहीं है. यह एलिमेंट, कम प्राथमिकता वाली मेनिफ़ेस्ट फ़ाइल से मिला होता है, जिसे कंट्रोल नहीं किया जा सकता. जैसे, इंपोर्ट की गई लाइब्रेरी.

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity-alias android:name="com.example.alias">
  <meta-data android:name="cow"
      android:value="@string/moo"/>
  <meta-data android:name="duck"
      android:value="@string/quack"/>
</activity-alias>

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity-alias android:name="com.example.alias">
  <meta-data android:name="cow"
      tools:node="remove"/>
</activity-alias>

मर्ज किए गए मेनिफ़ेस्ट का नतीजा:

<activity-alias android:name="com.example.alias">
  <meta-data android:name="duck"
      android:value="@string/quack"/>
</activity-alias>
tools:node="removeAll"
tools:node="remove" की तरह ही, लेकिन यह इस एलिमेंट टाइप से मैच करने वाले सभी एलिमेंट को हटा देता है (एक ही पैरंट एलिमेंट में).

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity-alias android:name="com.example.alias">
  <meta-data android:name="cow"
      android:value="@string/moo"/>
  <meta-data android:name="duck"
      android:value="@string/quack"/>
</activity-alias>

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity-alias android:name="com.example.alias">
  <meta-data tools:node="removeAll"/>
</activity-alias>

मर्ज किए गए मेनिफ़ेस्ट का नतीजा:

<activity-alias android:name="com.example.alias">
</activity-alias>
tools:node="replace"
कम प्राथमिकता वाले एलिमेंट को पूरी तरह से बदलें. इसका मतलब है कि अगर कम प्राथमिकता वाले मेनिफ़ेस्ट में मैच करने वाला कोई एलिमेंट है, तो उसे अनदेखा करें और इस एलिमेंट का इस्तेमाल ठीक वैसे ही करें जैसे वह इस मेनिफ़ेस्ट में दिखता है.

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity-alias android:name="com.example.alias">
  <meta-data android:name="cow"
      android:value="@string/moo"/>
  <meta-data android:name="duck"
      android:value="@string/quack"/>
</activity-alias>

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity-alias android:name="com.example.alias"
    tools:node="replace">
  <meta-data android:name="fox"
      android:value="@string/dingeringeding"/>
</activity-alias>

मर्ज किए गए मेनिफ़ेस्ट का नतीजा:

<activity-alias android:name="com.example.alias">
  <meta-data android:name="fox"
      android:value="@string/dingeringeding"/>
</activity-alias>
tools:node="strict"
जब भी कम प्राथमिकता वाले मेनिफ़ेस्ट में मौजूद यह एलिमेंट, ज़्यादा प्राथमिकता वाले मेनिफ़ेस्ट में मौजूद एलिमेंट से पूरी तरह मेल न खाता हो, तब बिल्ड में गड़बड़ी का मैसेज जनरेट करें. हालांकि, अगर इसे मर्ज करने के दूसरे नियम के मार्कर से ठीक कर दिया जाता है, तो ऐसा नहीं होगा. इससे, मर्ज करने से जुड़ी समस्याओं को हल करने के लिए इस्तेमाल होने वाले हेयुरिस्टिक्स की सेटिंग बदल जाती है. उदाहरण के लिए, अगर कम प्राथमिकता वाले मेनिफ़ेस्ट में कोई अतिरिक्त एट्रिब्यूट शामिल है, तो बिल्ड पूरा नहीं होता. हालांकि, डिफ़ॉल्ट व्यवहार के तहत, मर्ज किए गए मेनिफ़ेस्ट में अतिरिक्त एट्रिब्यूट जोड़ दिया जाता है.

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:windowSoftInputMode="stateUnchanged">
    <intent-filter>
        <action android:name="android.intent.action.SEND" />
        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>
</activity>

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:screenOrientation="portrait"
    tools:node="strict">
</activity>

इससे मेनिफ़ेस्ट मर्ज करने से जुड़ी गड़बड़ी होती है. स्ट्रिक्ट मोड में, दोनों मेनिफ़ेस्ट एलिमेंट एक जैसे होने चाहिए. इन अंतरों को हल करने के लिए, आपको मर्ज करने के अन्य नियम के मार्कर लागू करने होंगे. (tools:node="strict" के बिना, ये दोनों फ़ाइलें बिना किसी गड़बड़ी के मर्ज हो सकती हैं, जैसा कि tools:node="merge" के उदाहरण में दिखाया गया है.)

एट्रिब्यूट मार्कर

अगर आपको सिर्फ़ मेनिफ़ेस्ट टैग में मौजूद कुछ खास एट्रिब्यूट पर मर्ज करने का नियम लागू करना है, तो इन एट्रिब्यूट का इस्तेमाल करें. हर एट्रिब्यूट में एक या उससे ज़्यादा एट्रिब्यूट के नाम (एट्रिब्यूट नेमस्पेस के साथ) शामिल किए जा सकते हैं. इन्हें कॉमा लगाकर अलग किया जाता है.

tools:remove="attr, ..."
मर्ज किए गए मेनिफ़ेस्ट से, बताए गए एट्रिब्यूट हटाएं. इसका इस्तेमाल तब किया जाता है, जब कम प्राथमिकता वाली मेनिफ़ेस्ट फ़ाइल में ये एट्रिब्यूट शामिल हों और आपको यह पक्का करना हो कि ये एट्रिब्यूट, मर्ज किए गए मेनिफ़ेस्ट में शामिल न हों.

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:windowSoftInputMode="stateUnchanged">

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:screenOrientation="portrait"
    tools:remove="android:windowSoftInputMode">

मर्ज किए गए मेनिफ़ेस्ट का नतीजा:

<activity android:name="com.example.ActivityOne"
    android:screenOrientation="portrait">
tools:replace="attr, ..."
कम प्राथमिकता वाले मेनिफ़ेस्ट में मौजूद एट्रिब्यूट को, इस मेनिफ़ेस्ट में मौजूद एट्रिब्यूट से बदलें. दूसरे शब्दों में, हमेशा ज़्यादा प्राथमिकता वाले मेनिफ़ेस्ट की वैल्यू रखें.

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:theme="@oldtheme"
    android:exported="false"
    android:windowSoftInputMode="stateUnchanged">

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:theme="@newtheme"
    android:exported="true"
    android:screenOrientation="portrait"
    tools:replace="android:theme,android:exported">

मर्ज किए गए मेनिफ़ेस्ट का नतीजा:

<activity android:name="com.example.ActivityOne"
    android:theme="@newtheme"
    android:exported="true"
    android:screenOrientation="portrait"
    android:windowSoftInputMode="stateUnchanged">
tools:strict="attr, ..."
जब भी कम प्राथमिकता वाले मेनिफ़ेस्ट में मौजूद ये एट्रिब्यूट, ज़्यादा प्राथमिकता वाले मेनिफ़ेस्ट में मौजूद एट्रिब्यूट से पूरी तरह मेल न खाते हों, तब बिल्ड में गड़बड़ी का मैसेज जनरेट करें. यह सभी एट्रिब्यूट के लिए डिफ़ॉल्ट व्यवहार है. हालांकि, मर्ज करने से जुड़ी समस्याओं को हल करने के लिए इस्तेमाल होने वाले हेयुरिस्टिक्स में बताए गए खास व्यवहार वाले एट्रिब्यूट के लिए यह लागू नहीं होता.

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:screenOrientation="landscape">
</activity>

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:screenOrientation="portrait"
    tools:strict="android:screenOrientation">
</activity>

इससे मेनिफ़ेस्ट मर्ज करने से जुड़ी गड़बड़ी होती है. समस्या को हल करने के लिए, आपको मर्ज करने के दूसरे नियम के मार्कर लागू करने होंगे. यह डिफ़ॉल्ट रूप से होता है. इसलिए, tools:strict="screenOrientation" को साफ़ तौर पर जोड़ने पर भी यही नतीजा मिलता है.

एक एलिमेंट पर कई मार्कर भी लागू किए जा सकते हैं, जैसा कि इस उदाहरण में दिखाया गया है:

कम प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:theme="@oldtheme"
    android:exported="false"
    android:allowTaskReparenting="true"
    android:windowSoftInputMode="stateUnchanged">

ज़्यादा प्राथमिकता वाला मेनिफ़ेस्ट:

<activity android:name="com.example.ActivityOne"
    android:theme="@newtheme"
    android:exported="true"
    android:screenOrientation="portrait"
    tools:replace="android:theme,android:exported"
    tools:remove="android:windowSoftInputMode">

मर्ज किए गए मेनिफ़ेस्ट का नतीजा:

<activity android:name="com.example.ActivityOne"
    android:theme="@newtheme"
    android:exported="true"
    android:allowTaskReparenting="true"
    android:screenOrientation="portrait">

मार्कर सिलेक्टर

अगर आपको मर्ज करने के नियम के मार्कर सिर्फ़ किसी एक इंपोर्ट की गई लाइब्रेरी पर लागू करने हैं, तो लाइब्रेरी पैकेज के नाम के साथ tools:selector एट्रिब्यूट जोड़ें.

उदाहरण के लिए, नीचे दिए गए मेनिफ़ेस्ट में, remove मर्ज करने का नियम सिर्फ़ तब लागू होता है, जब कम प्राथमिकता वाली मेनिफ़ेस्ट फ़ाइल, com.example.lib1 लाइब्रेरी से हो:

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

अगर कम प्राथमिकता वाला मेनिफ़ेस्ट किसी दूसरे सोर्स से है, तो remove मर्ज करने के नियम को अनदेखा कर दिया जाता है.

ध्यान दें: अगर इसका इस्तेमाल किसी एट्रिब्यूट मार्कर के साथ किया जाता है, तो यह मार्कर में बताए गए सभी एट्रिब्यूट पर लागू होता है.

इंपोर्ट की गई लाइब्रेरी के लिए <uses-sdk> को बदलना

डिफ़ॉल्ट रूप से, मुख्य मेनिफ़ेस्ट फ़ाइल से ज़्यादा minSdk वैल्यू वाली लाइब्रेरी इंपोर्ट करने पर, गड़बड़ी होती है और लाइब्रेरी इंपोर्ट नहीं की जा सकती.

मर्ज टूल को इस विरोध को अनदेखा करने और आपके ऐप्लिकेशन की कम minSdk वैल्यू को बनाए रखते हुए लाइब्रेरी इंपोर्ट करने के लिए, <uses-sdk> टैग में overrideLibrary एट्रिब्यूट जोड़ें. एट्रिब्यूट की वैल्यू, एक या एक से ज़्यादा लाइब्रेरी पैकेज के नाम हो सकते हैं. इन्हें कॉमा लगाकर अलग किया जाता है. इससे उन लाइब्रेरी के बारे में पता चलता है जो मुख्य मेनिफ़ेस्ट के minSdk को बदल सकती हैं.

उदाहरण के लिए, अगर आपके ऐप्लिकेशन का मुख्य मेनिफ़ेस्ट इस तरह overrideLibrary लागू करता है:

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
          package="com.example.app"
          xmlns:tools="http://schemas.android.com/tools">
  <uses-sdk tools:overrideLibrary="com.example.lib1, com.example.lib2"/>
...

इसके बाद, <uses-sdk> टैग से जुड़ी गड़बड़ी के बिना, नीचे दिए गए मेनिफ़ेस्ट को मर्ज किया जा सकता है. साथ ही, मर्ज किए गए मेनिफ़ेस्ट में ऐप्लिकेशन मेनिफ़ेस्ट से minSdk="2" को हटा दिया जाता है.

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
          package="com.example.lib1">
   <uses-sdk android:minSdk="4" />
...

सिस्टम की अनुमतियां

Android के कुछ एपीआई, पहले ऐप्लिकेशन के लिए आसानी से ऐक्सेस किए जा सकते थे. हालांकि, Android के हाल ही के वर्शन में, सिस्टम की अनुमतियों की वजह से इन पर पाबंदी लगा दी गई है.

जिन ऐप्लिकेशन को इन एपीआई का ऐक्सेस चाहिए उन्हें काम करने में आने वाली समस्याओं से बचाने के लिए, Android के नए वर्शन में ऐप्लिकेशन को अनुमति के बिना भी उन एपीआई को ऐक्सेस करने की अनुमति दी जाती है. हालांकि, ऐसा तब ही होता है, जब targetSdkVersion को उस वर्शन से कम वैल्यू पर सेट किया गया हो जिसमें पाबंदी जोड़ी गई थी. इस तरह के व्यवहार से, ऐप्लिकेशन को एपीआई को ऐक्सेस करने की अनुमति मिल जाती है. जिन मर्ज किए गए मेनिफ़ेस्ट में targetSdkVersion के लिए अलग-अलग वैल्यू हैं उन पर असर पड़ सकता है.

अगर कम प्राथमिकता वाली मेनिफ़ेस्ट फ़ाइल में targetSdkVersion के लिए कम वैल्यू है, जिससे उसे अनुमति मिलती है और ज़्यादा प्राथमिकता वाली मेनिफ़ेस्ट में ऐसी अनुमति नहीं है (क्योंकि इसकी targetSdkVersion वैल्यू, उस वर्शन के बराबर या उससे ज़्यादा है जिसमें पाबंदी जोड़ी गई थी), तो मर्ज करने वाला टूल, मर्ज किए गए मेनिफ़ेस्ट में सिस्टम की अनुमति को साफ़ तौर पर जोड़ता है.

उदाहरण के लिए, अगर आपके ऐप्लिकेशन में targetSdkVersion को 4 या उससे ज़्यादा पर सेट किया गया है और targetSdkVersion को 3 या उससे कम पर सेट करके कोई लाइब्रेरी इंपोर्ट की गई है, तो मर्ज करने वाला टूल, मर्ज किए गए मेनिफ़ेस्ट में WRITE_EXTERNAL_STORAGE अनुमति जोड़ता है.

टेबल 2 में, उन सभी अनुमतियों की सूची दी गई है जिन्हें आपके मर्ज किए गए मेनिफ़ेस्ट में जोड़ा जा सकता है:

टेबल 2. उन अनुमतियों की सूची जिन्हें मर्ज करने वाला टूल, मर्ज किए गए मेनिफ़ेस्ट में जोड़ सकता है

कम प्राथमिकता वाले मेनिफ़ेस्ट में बताए गए एलान मर्ज किए गए मेनिफ़ेस्ट में जोड़ी गई अनुमतियां
targetSdkVersion 3 या उससे कम है WRITE_EXTERNAL_STORAGE, READ_PHONE_STATE
targetSdkVersion 15 या उससे पहले का वर्शन है और READ_CONTACTS का इस्तेमाल किया जा रहा है READ_CALL_LOG
targetSdkVersion 15 या उससे पहले का वर्शन है और WRITE_CONTACTS का इस्तेमाल किया जा रहा है WRITE_CALL_LOG

मर्ज किए गए मेनिफ़ेस्ट की जांच करना और उसमें मौजूद समस्याओं का पता लगाना

ऐप्लिकेशन बनाने से पहले ही, मर्ज किए गए मेनिफ़ेस्ट की झलक देखी जा सकती है. झलक देखने के लिए, यह तरीका अपनाएं:

  1. Android Studio में, अपनी AndroidManifest.xml फ़ाइल खोलें.
  2. एडिटर में सबसे नीचे मौजूद, मर्ज किया गया मेनिफ़ेस्ट टैब पर क्लिक करें.

मर्ज किए गए मेनिफ़ेस्ट व्यू में, बाईं ओर मर्ज किए गए मेनिफ़ेस्ट के नतीजे दिखते हैं और दाईं ओर, मर्ज की गई हर मेनिफ़ेस्ट फ़ाइल की जानकारी दिखती है, जैसा कि दूसरे चित्र में दिखाया गया है.

कम प्राथमिकता वाली मेनिफ़ेस्ट फ़ाइलों से मर्ज किए गए एलिमेंट, बाईं ओर अलग-अलग रंगों में हाइलाइट किए गए हैं. हर रंग की कुंजी, मेनिफ़ेस्ट सोर्स में बताई गई है.

दूसरी इमेज. मर्ज किए गए मेनिफ़ेस्ट का व्यू.

जिन मेनिफ़ेस्ट फ़ाइलों को बिल्ड में शामिल किया गया था, लेकिन जिनमें एलिमेंट या एट्रिब्यूट शामिल नहीं थे उन्हें अन्य मेनिफ़ेस्ट फ़ाइलें में शामिल किया जाता है.

किसी एलिमेंट के सोर्स के बारे में जानकारी देखने के लिए, बाएं पैनल में उस पर क्लिक करें. इसके बाद, मर्जिंग लॉग में जानकारी दिखेगी.

अगर कोई विरोध होता है, तो वह मर्ज करने से जुड़ी गड़बड़ियां में दिखता है. साथ ही, मर्ज करने के नियम के मार्कर का इस्तेमाल करके, विरोध को हल करने का सुझाव भी दिखता है.

गड़बड़ियां, इवेंट लॉग विंडो में भी प्रिंट होती हैं. इन्हें देखने के लिए, व्यू > टूल विंडो > इवेंट लॉग चुनें.

मर्ज किए जा रहे डिसीज़न ट्री का पूरा लॉग देखने के लिए, लॉग फ़ाइल को अपने मॉड्यूल की build/outputs/logs/ डायरेक्ट्री में manifest-merger-buildVariant-report.txt नाम से ढूंढें.

नीतियां मर्ज करना

मेनिफ़ेस्ट मर्ज करने वाला टूल, एक मेनिफ़ेस्ट फ़ाइल के हर एक्सएमएल एलिमेंट को, दूसरी फ़ाइल के मिलते-जुलते एलिमेंट से मैच कर सकता है. मर्ज करने की सुविधा, मैच की सेटिंग का इस्तेमाल करके हर एलिमेंट से मैच करती है. यह सेटिंग, एट्रिब्यूट की यूनीक वैल्यू (जैसे, android:name) या टैग की यूनीक वैल्यू (उदाहरण के लिए, सिर्फ़ एक <supports-screen> एलिमेंट हो सकता है) हो सकती है.

अगर दो मेनिफ़ेस्ट में एक ही एक्सएमएल एलिमेंट है, तो टूल, मर्ज करने की तीन नीतियों में से किसी एक का इस्तेमाल करके, दोनों एलिमेंट को आपस में मर्ज कर देता है:

मर्ज करें
एक-दूसरे से मेल न खाने वाले सभी एट्रिब्यूट को एक ही टैग में जोड़ें और चाइल्ड एलिमेंट को, मर्ज करने की उनकी नीति के मुताबिक मर्ज करें. अगर कोई एट्रिब्यूट एक-दूसरे से मेल नहीं खाता है, तो मर्ज करने के नियम के मार्कर की मदद से, उन्हें एक साथ मर्ज करें.
सिर्फ़ चाइल्ड एंट्री मर्ज करना
एट्रिब्यूट को आपस में न जोड़ें या मर्ज न करें. सिर्फ़ सबसे ज़्यादा प्राथमिकता वाली मेनिफ़ेस्ट फ़ाइल से मिले एट्रिब्यूट रखें. साथ ही, एलिमेंट को मर्ज करने की नीति के मुताबिक, चाइल्ड एलिमेंट को मर्ज करें.
Keep
एलिमेंट को वैसे ही रहने दें और उसे मर्ज की गई फ़ाइल में मौजूद कॉमन पैरंट एलिमेंट में जोड़ें. इसका इस्तेमाल सिर्फ़ तब किया जाता है, जब एक ही एलिमेंट के कई एलान किए जा सकते हों.

टेबल 3 में हर एलिमेंट टाइप, इस्तेमाल की गई मर्ज नीति का टाइप, और दो मेनिफ़ेस्ट के बीच एलिमेंट मैच का पता लगाने के लिए इस्तेमाल की गई कुंजी की सूची दी गई है:

टेबल 3. मेनिफ़ेस्ट एलिमेंट को मर्ज करने की नीतियां और मैच करने वाली कुंजियां

कौन-कौन सी चीज़ें शामिल हैं मर्ज करने की नीति मैच की
<action> मर्ज करें android:name एट्रिब्यूट
<activity> मर्ज करें android:name एट्रिब्यूट
<application> मर्ज करें हर <manifest> के लिए सिर्फ़ एक है.
<category> मर्ज करें android:name एट्रिब्यूट
<data> मर्ज करें हर <intent-filter> के लिए सिर्फ़ एक है.
<grant-uri-permission> मर्ज करें हर <provider> के लिए सिर्फ़ एक है.
<instrumentation> मर्ज करें android:name एट्रिब्यूट
<intent-filter> Keep कोई मैच नहीं; पैरंट एलिमेंट में कई एलान किए जा सकते हैं.
<manifest> सिर्फ़ चाइल्ड एंट्री मर्ज करना हर फ़ाइल में सिर्फ़ एक होता है.
<meta-data> मर्ज करें android:name एट्रिब्यूट
<path-permission> मर्ज करें हर <provider> के लिए सिर्फ़ एक है.
<permission-group> मर्ज करें android:name एट्रिब्यूट
<permission> मर्ज करें android:name एट्रिब्यूट
<permission-tree> मर्ज करें android:name एट्रिब्यूट
<provider> मर्ज करें android:name एट्रिब्यूट
<receiver> मर्ज करें android:name एट्रिब्यूट
<screen> मर्ज करें android:screenSize एट्रिब्यूट
<service> मर्ज करें android:name एट्रिब्यूट
<supports-gl-texture> मर्ज करें android:name एट्रिब्यूट
<supports-screen> मर्ज करें हर <manifest> के लिए सिर्फ़ एक है.
<uses-configuration> मर्ज करें हर <manifest> के लिए सिर्फ़ एक है.
<uses-feature> मर्ज करें android:name एट्रिब्यूट (अगर मौजूद नहीं है, तो android:glEsVersion एट्रिब्यूट)
<uses-library> मर्ज करें android:name एट्रिब्यूट
<uses-permission> मर्ज करें android:name एट्रिब्यूट
<uses-sdk> मर्ज करें हर <manifest> के लिए सिर्फ़ एक है.
कस्टम एलिमेंट मर्ज करें मेल नहीं खाते; ये मर्ज करने वाले टूल के लिए अनजान होते हैं और इन्हें हमेशा मर्ज किए गए मेनिफ़ेस्ट में शामिल किया जाता है.

मेनिफ़ेस्ट में बिल्ड वैरिएबल इंजेक्ट करना

अगर आपको अपनी AndroidManifest.xml फ़ाइल में ऐसे वैरिएबल डालने हैं जो आपकी build.gradle फ़ाइल में तय किए गए हैं, तो ऐसा manifestPlaceholders प्रॉपर्टी की मदद से किया जा सकता है. यह प्रॉपर्टी, की-वैल्यू पेयर का एक मैप लेती है, जैसा कि यहां दिखाया गया है:

Groovy

android {
    defaultConfig {
        manifestPlaceholders = [hostName:"www.example.com"]
    }
    ...
}

Kotlin

android {
    defaultConfig {
        manifestPlaceholders["hostName"] = "www.example.com"
    }
    ...
}

इसके बाद, किसी एक प्लेसहोल्डर को एट्रिब्यूट की वैल्यू के तौर पर मेनिफ़ेस्ट फ़ाइल में डाला जा सकता है:

<intent-filter ... >
    <data android:scheme="https" android:host="${hostName}" ... />
    ...
</intent-filter>

डिफ़ॉल्ट रूप से, बिल्ड टूल ${applicationId} प्लेसहोल्डर में आपके ऐप्लिकेशन का ऐप्लिकेशन आईडी भी उपलब्ध कराते हैं. यह वैल्यू, मौजूदा बिल्ड के लिए हमेशा फ़ाइनल ऐप्लिकेशन आईडी से मैच होती है. इसमें बिल्ड के वैरिएंट के हिसाब से किए गए बदलाव भी शामिल होते हैं. यह तब काम आता है, जब आपको अपने बिल्ड वैरिएंट के बीच भी, इंटेंट ऐक्शन जैसे आइडेंटिफ़ायर के लिए यूनीक नेमस्पेस का इस्तेमाल करना हो.

उदाहरण के लिए, अगर आपकी build.gradle फ़ाइल कुछ ऐसी दिखती है:

Groovy

android {
    defaultConfig {
        applicationId "com.example.myapp"
    }
    flavorDimensions "type"
    productFlavors {
        free {
            applicationIdSuffix ".free"
            dimension "type"
        }
        pro {
            applicationIdSuffix ".pro"
            dimension "type"
        }
    }
}

Kotlin

android {
    defaultConfig {
        applicationId = "com.example.myapp"
    }
    flavorDimensions += "type"
    productFlavors {
        create("free") {
            applicationIdSuffix = ".free"
            dimension = "type"
        }
        create("pro") {
            applicationIdSuffix = ".pro"
            dimension = "type"
        }
    }
}

इसके बाद, अपने मेनिफ़ेस्ट में ऐप्लिकेशन आईडी को इस तरह डाला जा सकता है:

<intent-filter ... >
    <action android:name="${applicationId}.TRANSMOGRIFY" />
    ...
</intent-filter>

"मुफ़्त" प्रॉडक्ट फ़्लेवर बनाने पर, मेनिफ़ेस्ट का नतीजा यह होता है:

<intent-filter ... >
   <action android:name="com.example.myapp.free.TRANSMOGRIFY" />
    ...
</intent-filter>

ज़्यादा जानकारी के लिए, ऐप्लिकेशन आईडी सेट करना लेख पढ़ें.