Android Studio错误:“Manifest merger failed: Apps targeting Android 12”;

我已经将我的模拟器版本和Android SDK版本更新到Android S (Android 12)。更新后,我无法运行项目。我不能运行< a href = " https://en.wikipedia.org/wiki/%22Hello _World !%22_program" rel="noreferrer">Hello, World!< / >项目(空项目),但我可以构建Gradle以及,但我不能运行项目。我总是得到错误:

Manifest合并失败:针对Android 12及更高版本的应用 时,必须为android: exported指定显式值 相应的组件定义了一个意图过滤器。看到 https://developer.android.com/guide/topics/manifest/activity-element#exported 对细节。< / p >

我该怎么解决呢?

以下是截图:

This is a截图。 < / >

使用Android 12 SDK时如何解决此问题?

这个问题是关于应用这个解决方案后的问题,与这个问题不同。而且,这个问题比更早。

289354 次浏览

你需要指定android:exported="false"android:exported="true"

清单:

<activity
android:name=".MainActivity"
android:exported="true"
android:theme="@style/Theme.MyApplication.NoActionBar">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>

文档所述:

如果你的应用目标是Android 12并且包含活动、服务或 广播接收器使用意图过滤器,你必须显式 为这些应用组件声明android: exported属性

警告:如果活动、服务或广播接收器使用intent 的过滤器,并且没有显式声明的值 Android:导出,您的应用程序不能安装在运行的设备上 Android 12。< / p >

同样检查当为'android:exported'值使用true/false时。

你的问题可能因为这篇文章而被标记为重复:针对Android 12的Manifest合并失败,尽管你的问题是在一周前发布的。我现在看不到国旗了。

为了澄清另一个答案,请注意,android:出口应该为你的主活动设置为true,否则它将不会启动,尽管来自Android Studio的鼓励“启动成功”消息,因为没有其他应用程序,甚至Android系统本身可以启动它。

<activity
android:name=".MainActivity"
android:exported="true"

对于其他意图隐藏在合并清单中的活动,这通常会被设置为false。

在清单中,添加android:出口=“true"或android:exported="false "在默认的启动活动属性中。

完成了!你可以在Android 12上运行你的应用程序。

<manifest ... >


<activity
android:name=".ui.dashboard.DashboardActivity"
android:screenOrientation="portrait"
android:exported="true"
android:theme="@style/AppTheme.Launcher">
<intent-filter>
<action android:name="android.intent.action.MAIN" />


<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</manifest>

根据您的需求设置android:出口值。

广播接收器是否可以从其应用程序之外的非系统源接收消息-“true”;如果可以,并且“虚假”;如果不是。如果“false”,广播接收器只能接收由系统、相同应用程序的组件或具有相同用户ID的应用程序发送的消息。

如果未指定,默认值取决于广播接收器是否包含意图过滤器。如果接收端包含至少一个意图过滤器,那么默认值是"true"。否则,默认值为&;false&;。

此属性不是限制广播接收器外部曝光的唯一方法。您还可以使用权限限制可以发送消息的外部实体(请参阅permission属性)。

安卓系统文档

我还必须将android:exported="true"添加到清单中声明的所有__abc1中。所以我有这样的东西:

<receiver android:name=".alarms.AlarmReScheduler"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.LOCKED_BOOT_COMPLETED" />
<action android:name="android.intent.action.BOOT_COMPLETED" />
<action android:name="android.intent.action.QUICKBOOT_POWERON" />
<action android:name="android.intent.action.PACKAGE_REPLACED" />


<!-- For HTC devices -->
<action android:name="com.htc.intent.action.QUICKBOOT_POWERON" />
</intent-filter>
</receiver>

我认为你不应该把android:exported="true"加到所有东西上。你应该只在那些Broadcast接收器需要对Android OS可见时添加它。这段代码中的Intent过滤器意味着我想让Android OS唤醒我的Android应用程序并执行一个操作。

android.intent.action.BOOT_COMPLETED是一个很好的例子,因为Android操作系统向安装在设备中的每个应用程序发送广播。因此,从技术上讲,这意味着任何具有带有动作的意图过滤器的Broadcast接收器都应该声明android:exported="true"

在我的项目瞄准Android 12后,我也遇到了同样的问题。

问题是该项目相当大,有多个AndroidManifest.xml文件,并且在许多地方缺少android:exported

我最终创建了一个Gradle任务来自动为我填充缺少的android:exported属性。

这里是链接

Enter image description here

在你的启动活动中,声明&;android: exported&;

<activity android:name=".MainActivity"
android:exported = "false">


<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>


</activity>

如果“android: exported= true”为真,则意味着任何应用程序都可以访问该活动,并且可以通过其确切的类名启动。

如果“android: exported = false”为false,则意味着该活动只能由具有相同用户ID的同一应用程序的组件或特权系统组件启动。

更多细节请查看在这里

如果您在清单中没有找到没有“android: exported = false”标签的活动;那么它很可能在你的依赖中…为了精确定位,首先降级“compileSdkVersion"targetSdkVersion"到30,所以它建立。

android {
compileSdkVersion("android-S")
buildToolsVersion "30.0.3"


defaultConfig {
...
minSdkVersion 23
targetSdkVersion("S")
...
}

在此之后,在主manifest.xml窗口中有一个带有“合并清单”的选项卡。在那里,你可以检查哪些活动没有“android: exported = false”;属性。

在我的案例中,这是因为第三方工具:

文件build.gradle (: app):

debugImplementation 'com.squareup.leakcanary:leakcanary-android:2.7'
//and
debugImplementation "com.github.markzhai:blockcanary-android:1.5.0"
releaseImplementation "com.github.markzhai:blockcanary-no-op:1.5.0"

此外,对于服务,我必须添加属性:

<service
android:name=".autofillservice.MyAutofillService"
android:exported="true"
android:permission="android.permission.BIND_AUTOFILL">

而且

<service
android:name="com.demo.myApp.my_access.MyAccessService"
android:enabled="true"
android:exported="true"
android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE">

由于我的问题是在第三方依赖,它不会很快更新,我只是添加了一个<活动>带有android标志的声明:exported="true"和出口=“false"在需要覆盖初始声明的地方,也因为我在调试中需要这个依赖,所以我在src/ Debug中添加了一个新的AndroidManifest.xml文件:

leak_canary:

<?xml version="1.0" encoding="UTF-8"?>


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


<application>
<activity
android:name="leakcanary.internal.activity.LeakActivity"
android:exported="true"
android:icon="@mipmap/leak_canary_icon"
android:label="@string/leak_canary_display_activity_label"
android:taskAffinity="com.squareup.leakcanary.${applicationId}"
android:theme="@style/leak_canary_LeakCanary.Base">


<intent-filter android:label="@string/leak_canary_import_hprof_file">


<action android:name="android.intent.action.VIEW" />


<category android:name="android.intent.category.DEFAULT" />
<category android:name="android.intent.category.BROWSABLE" />


<data android:scheme="file" />
<data android:scheme="content" />
<data android:mimeType="*/*" />
<data android:host="*" />


<data android:pathPattern=".*\\.hprof" />
<data android:pathPattern=".*\\..*\\.hprof" />
<data android:pathPattern=".*\\..*\\..*\\.hprof" />
<data android:pathPattern=".*\\..*\\..*\\..*\\.hprof" />
<data android:pathPattern=".*\\..*\\..*\\..*\\..*\\.hprof" />
<data android:pathPattern=".*\\..*\\..*\\..*\\..*\\..*\\.hprof" />
<data android:pathPattern=".*\\..*\\..*\\..*\\..*\\..*\\..*\\.hprof" />
</intent-filter>
</activity>


<activity
android:name="leakcanary.internal.RequestStoragePermissionActivity"
android:excludeFromRecents="true"
android:exported="false"
android:icon="@mipmap/leak_canary_icon"
android:label="@string/leak_canary_storage_permission_activity_label"
android:taskAffinity="com.squareup.leakcanary.${applicationId}"
android:theme="@style/leak_canary_Theme.Transparent" />


<receiver
android:name="leakcanary.internal.NotificationReceiver"
android:exported="false" />


</application>
</manifest>

你还不如直接使用这些工具:node="merge"属性,并将android:exported=true|false声明为LeoFarage好心建议

我得到这个错误,即使我添加android:exported="true"在所有活动,接收器等。对我有用的是将compileSdkVersiontargetSdkVersion改为30。

androidTestImplementation 'androidx.test.ext:junit:1.1.1'的版本更新为最新版本,如:

androidTestImplementation 'androidx.test.ext:junit:1.1.3'build.gradle应用程序级别。

针对Android 12的应用

将应用程序的targetSdkVersion更改为< >强S < / >强(32或31)以启用新行为。

然后根据活动将Manifest中的android:出口=“;“属性指定为true或false。

对于启动器活动,如splash或MainActivity,使用android:出口=“true",对于其余的活动,使用android:出口=“false"

例如:

<!-- It's **true** for the launcher Activity -->
<activity android:name=".SplashActivity"
android:exported="true" />


<!-- It's **false** for the rest of the Activities -->
<activity android:name=".MainActivity"
android:exported="false" />

笔记

启动器活动所需

<activity
android:name=".MainActivity"
android:exported="true"
android:theme="@style/Theme.MyApplication.NoActionBar">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>

接收方要求

<receiver android:name=".Musicreceiver"
android:exported="true">


</receiver>

服务要求

<service
android:name=".service.LoggerService"
android:enabled="true" />

对我来说,这两种解决方案都不起作用,即使我反复检查了“合并宣言”。在Android Studio。在编译项目后,错误就出现了,我无法识别生成问题的行。

解决方案:确保你的目标是项目中的最新库。我使用了团结, StartApp和Flurry Analytics。我忘记更新这些库,升级后,错误消失了。看起来这些库使用了来自 之前SDK 31 . < / p >

如果您正在使用废弃的库,那么您应该考虑替换它们。

添加android:出口=“true":

<receiver android:name="com.dieam.reactnativepushnotification.modules.RNPushNotificationActions"/>
<receiver android:name="com.dieam.reactnativepushnotification.modules.RNPushNotificationPublisher"/>
<receiver android:name="com.dieam.reactnativepushnotification.modules.RNPushNotificationBootEventReceiver"
android:exported="true">


<intent-filter>
<action android:name="android.intent.action.BOOT_COMPLETED" />
<action android:name="android.intent.action.QUICKBOOT_POWERON" />
<action android:name="com.htc.intent.action.QUICKBOOT_POWERON"/>
</intent-filter>
</receiver>