检查应用程序是否在前台或后台运行(使用同步适配器)

5
我知道这是一个重复的问题,但我已经到处寻找,并没有找到适合我的解决方案,所以...
我有一个应用程序,从TMDB API获取电影数据,它使用同步适配器按页获取数据,基本上它运行得很好,除非在打开应用程序时进行同步适配器的周期性同步且用户不在第一页。所以我的选择是强制更新电影列表并将用户发送到列表顶部,这是非常糟糕的体验,所以不能作为选项考虑,或者我可以检查应用程序是否正在运行,无论是在前台还是在应用程序堆栈中,它对用户不可见但仍在运行。所以我通过搜索最好能得到这段代码:
   public static boolean isAppRunning(final Context context, final String packageName) {

    final ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
    final List<ActivityManager.RunningAppProcessInfo> procInfos = activityManager.getRunningAppProcesses();

    if (procInfos != null) {
        for (final ActivityManager.RunningAppProcessInfo processInfo : procInfos) {
            if (processInfo.processName.equals(packageName))
                return true;
        }
    }

    return false;
}

但由于某些原因,它不能正常工作,并且即使我通过滑动杀死应用程序也仍然将应用程序视为正在运行。起初我认为这是因为我有一个同步适配器可以被视为应用程序正在运行,因此我在清单中使用了另一个进程进行处理。

android:process=":service"

但它没有起作用。

我也考虑在活动的onStart和onStop中使用变量,但不能保证当应用程序关闭或被杀死时会调用onStop,因此尝试避免使用这种方法。

这是我的manifest.xml文件:

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

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="inc.mourad.ahmed.watchme">

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.AUTHENTICATE_ACCOUNTS" />
<uses-permission android:name="android.permission.READ_SYNC_SETTINGS" />
<uses-permission android:name="android.permission.WRITE_SYNC_SETTINGS" />

<permission
    android:name="inc.ahmed.mourad.ACCESS_WATCHME_DATABASE"
    android:label="access my movies content provider"
    android:protectionLevel="dangerous" />
<permission
    android:name="inc.ahmed.mourad.ACCESS_WATCHME_SYNC_SERVICE"
    android:label="access my sync service"
    android:protectionLevel="dangerous" />
<permission
    android:name="inc.ahmed.mourad.ACCESS_WATCHME_SYNC_AUTHENTICATOR_SERVICE"
    android:label="access my sync authenticator dummy service"
    android:protectionLevel="dangerous" />

<application
    android:allowBackup="true"
    android:icon="@mipmap/ic_launcher"
    android:label="@string/app_name"
    android:supportsRtl="true"
    android:theme="@style/Theme.AppCompat">

    <activity android:name=".SplashScreenActivity"
        android:theme="@style/Theme.AppCompat.NoActionBar">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />

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

    <activity
        android:name=".MainActivity"
        android:label="@string/app_name"
        android:theme="@style/Theme.AppCompat">
        <intent-filter>
            <data
                android:host="www.ahmedmourad.com"
                android:pathPrefix="/watchme/"
                android:scheme="http" />

            <action android:name="android.intent.action.VIEW" />
            <category android:name="android.intent.category.DEFAULT"/>
            <category android:name="android.intent.category.BROWSABLE"/>
        </intent-filter>
    </activity>

    <activity
        android:name=".SettingsActivity"
        android:label="@string/title_activity_settings"
        android:parentActivityName=".MainActivity">
        <meta-data
            android:name="android.support.PARENT_ACTIVITY"
            android:value="inc.mourad.ahmed.watchme.MainActivity" />
    </activity>
    <activity android:name=".MovieDetailsActivity" />

    <provider
        android:name=".data.MovieProvider"
        android:authorities="@string/content_authority"
        android:enabled="true"
        android:exported="false"
        android:permission="inc.ahmed.mourad.ACCESS_WATCHME_DATABASE"
        android:syncable="true" />

    <!-- SyncAdapter's dummy authentication service -->
    <service android:name=".sync.WatchMeAuthenticatorService"
        android:permission="inc.ahmed.mourad.ACCESS_WATCHME_SYNC_AUTHENTICATOR_SERVICE"
        android:process=":service">
        <intent-filter>
            <action android:name="android.accounts.AccountAuthenticator" />
        </intent-filter>

        <meta-data
            android:name="android.accounts.AccountAuthenticator"
            android:resource="@xml/authenticator" />
    </service>

    <!-- The SyncAdapter service -->
    <service
        android:name=".sync.WatchMeSyncService"
        android:exported="true"
        android:permission="inc.ahmed.mourad.ACCESS_WATCHME_SYNC_SERVICE"
        android:process=":service">
        <intent-filter>
            <action android:name="android.content.SyncAdapter" />
        </intent-filter>

        <meta-data
            android:name="android.content.SyncAdapter"
            android:resource="@xml/syncadapter" />
    </service>

</application>

那么,有什么想法吗?提前感谢。


onStop方法在API 11及以上版本中保证会被调用,这难道不够了吗? - X3Btel
@X3Btel,这里说在低内存情况下可能不会被调用,因此我正在寻找更安全的解决方案。如果没有,我将继续使用它并尝试减少损失。 - Ahmed Mourad
我使用了这个答案 https://dev59.com/OG855IYBdhLWcg3wUSgW#19920353 它运行良好,唯一的问题是如果你锁定/解锁手机,但在你的情况下应该也能工作。 - X3Btel
@X3Btel 没有注意到那一点,好的会去做,谢谢你的帮助。 - Ahmed Mourad
谢谢你的另一个回答,这里肯定需要它。 - Ahmed Mourad
显示剩余2条评论
2个回答

16

我们对您的想法进行了微调,并提出了以下Kotlin代码(有关Java,请参见下文):

fun appInForeground(context: Context): Boolean {
    val activityManager = context.getSystemService(Context.ACTIVITY_SERVICE) as ActivityManager
    val runningAppProcesses = activityManager.runningAppProcesses ?: return false
    return runningAppProcesses.any { it.processName == context.packageName && it.importance == ActivityManager.RunningAppProcessInfo.IMPORTANCE_FOREGROUND }
}

这是 Java 的等效代码:

private boolean appInForeground(@NonNull Context context) {
    ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
    List<ActivityManager.RunningAppProcessInfo> runningAppProcesses = activityManager.getRunningAppProcesses();
    if (runningAppProcesses == null) {
        return false;
    }

    for (ActivityManager.RunningAppProcessInfo runningAppProcess : runningAppProcesses) {
        if (runningAppProcess.processName.equals(context.getPackageName()) &&
                runningAppProcess.importance == ActivityManager.RunningAppProcessInfo.IMPORTANCE_FOREGROUND) {
            return true;
        }
    }
    return false;
}

这是相关文档。


5

原始答案:https://dev59.com/KHA65IYBdhLWcg3wuhIR#48767617 按照Android文档推荐的方法进行操作

class MyApplication : Application(), LifecycleObserver {

override fun onCreate() {
    super.onCreate()
    ProcessLifecycleOwner.get().lifecycle.addObserver(this)
}

fun isInForeground(): Boolean {
    return ProcessLifecycleOwner.get().lifecycle.currentState.isAtLeast(Lifecycle.State.STARTED)
}

@OnLifecycleEvent(Lifecycle.Event.ON_STOP)
fun onAppBackgrounded() {
    //App in background

    Log.e(TAG, "************* backgrounded")
    Log.e(TAG, "************* ${isInForeground()}")
}

@OnLifecycleEvent(Lifecycle.Event.ON_START)
fun onAppForegrounded() {

    Log.e(TAG, "************* foregrounded")
    Log.e(TAG, "************* ${isInForeground()}")
    // App in foreground
}}

在您的gradle(应用)中添加:implementation "androidx.lifecycle:lifecycle-extensions:2.2.0" 然后在运行时检查状态,请调用(applicationContext as MyApplication).isActivityVisible()

但我认为这还不是一个适当的解决方案。因为每次我导航到另一个活动时,onAppBackgrounded()都会针对当前活动调用,而我正在导航到下一个活动时会得到onAppForegrounded() - A S M Sayem
我无法复制这个问题,您是否只在应用程序类中注册了生命周期观察器?因为 onAppBackgrounded 只应在应用程序的生命周期停止时调用(@OnLifecycleEvent(Lifecycle.Event.ON_STOP))。您可以在Android文档中阅读更多相关信息。@Saadat - Yonko Kilasi

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接