【问题标题】:How to launch the app with a specific Activity only when the app is launched for the first time?如何仅在第一次启动应用程序时启动具有特定 Activity 的应用程序?
【发布时间】:2016-03-17 13:58:31
【问题描述】:

我只想在我的应用程序第一次启动时启动一个 Activity,从那时起,该应用程序每次都应该与另一个(启动器)Activity 一起启动。所以我实现了一个基于this SO answer的解决方案。

解决方案围绕boolean 首选项(以下代码中以startedBeforePreferenceKey 作为键的首选项)。在启动器活动的onCreate() 中,我尝试使用键startedBeforePreferenceKey 检索首选项并将其存储在变量startedBefore 中。如果首选项不存在,则将startedBefore 分配给false

然后我检查startedBefore 是否为false,如果是,我创建提到的首选项,给它一个值true 并将其存储在SharedPreferences,然后启动这个应该第一次启动应用程序的活动。这样,当下次执行 onCreate() 时执行此检查时,startedBefore 将被分配 true,因此此启动一次活动将不会启动。

问题是,当应用程序首次启动时,正常的启动器会在 Activity 之前立即显示,只有在应用程序首次启动时才应该启动应用程序已启动。

第一次启动应用时,正常的启动器根本不应该显示。直接,我想在第一次启动应用程序时显示的特殊活动,应该显示。

我该怎么办?

SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this);
boolean startedBefore = sharedPreferences.getBoolean(getString(R.string.startedBeforePreferenceKey), false);

if (!startedBefore) {
    SharedPreferences.Editor sharedPreferencesEditor = sharedPreferences.edit();
    sharedPreferencesEditor.putBoolean(getString(R.string.startedBeforePreferenceKey), true);
    sharedPreferencesEditor.commit();
    startActivity(new Intent(this, MainActivity.class)); 
} 

编辑: @HammadTariqSahi

首先,这段摘自 LogCat:

03-16 08:42:25.629: E/AndroidRuntime(1837): FATAL EXCEPTION: main
03-16 08:42:25.629: E/AndroidRuntime(1837): Process: tests.globalactivitytest, PID: 1837
03-16 08:42:25.629: E/AndroidRuntime(1837): java.lang.RuntimeException: Unable to instantiate application tests.globalactivitytest.activity.GlobalActivity: java.lang.ClassNotFoundException: Didn't find class "tests.globalactivitytest.activity.GlobalActivity" on path: DexPathList[[zip file "/data/app/tests.globalactivitytest-1/base.apk"],nativeLibraryDirectories=[/data/app/tests.globalactivitytest-1/lib/x86, /vendor/lib, /system/lib]]
03-16 08:42:25.629: E/AndroidRuntime(1837):     at android.app.LoadedApk.makeApplication(LoadedApk.java:578)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at android.app.ActivityThread.handleBindApplication(ActivityThread.java:4680)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at android.app.ActivityThread.-wrap1(ActivityThread.java)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1405)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at android.os.Handler.dispatchMessage(Handler.java:102)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at android.os.Looper.loop(Looper.java:148)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at android.app.ActivityThread.main(ActivityThread.java:5417)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at java.lang.reflect.Method.invoke(Native Method)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:726)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:616)
03-16 08:42:25.629: E/AndroidRuntime(1837): Caused by: java.lang.ClassNotFoundException: Didn't find class "tests.globalactivitytest.activity.GlobalActivity" on path: DexPathList[[zip file "/data/app/tests.globalactivitytest-1/base.apk"],nativeLibraryDirectories=[/data/app/tests.globalactivitytest-1/lib/x86, /vendor/lib, /system/lib]]
03-16 08:42:25.629: E/AndroidRuntime(1837):     at dalvik.system.BaseDexClassLoader.findClass(BaseDexClassLoader.java:56)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at java.lang.ClassLoader.loadClass(ClassLoader.java:511)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at java.lang.ClassLoader.loadClass(ClassLoader.java:469)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at android.app.Instrumentation.newApplication(Instrumentation.java:981)
03-16 08:42:25.629: E/AndroidRuntime(1837):     at android.app.LoadedApk.makeApplication(LoadedApk.java:573)
03-16 08:42:25.629: E/AndroidRuntime(1837):     ... 9 more
03-16 08:42:25.629: E/AndroidRuntime(1837):     Suppressed: java.lang.ClassNotFoundException: tests.globalactivitytest.activity.GlobalActivity
03-16 08:42:25.629: E/AndroidRuntime(1837):         at java.lang.Class.classForName(Native Method)
03-16 08:42:25.629: E/AndroidRuntime(1837):         at java.lang.BootClassLoader.findClass(ClassLoader.java:781)
03-16 08:42:25.629: E/AndroidRuntime(1837):         at java.lang.BootClassLoader.loadClass(ClassLoader.java:841)
03-16 08:42:25.629: E/AndroidRuntime(1837):         at java.lang.ClassLoader.loadClass(ClassLoader.java:504)
03-16 08:42:25.629: E/AndroidRuntime(1837):         ... 12 more
03-16 08:42:25.629: E/AndroidRuntime(1837):     Caused by: java.lang.NoClassDefFoundError: Class not found using the boot class loader; no stack trace available

GlobalActivity.java:

package tests.globalactivitytest;

import android.app.Application;
import android.content.Intent;
import android.content.SharedPreferences;
import android.preference.PreferenceManager;

public class GlobalActivity extends Application {
    @Override
    public void onCreate() {
        super.onCreate();

        SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
        //SharedPreferences.Editor editor = sharedPreferences.edit(); 
        boolean launchedBefore = sharedPreferences.getBoolean("launchedBefore", false);
        if (launchedBefore) {
            Intent intent = new Intent(this, MainActivity.class);
            intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            startActivity(intent);
        } else {
            Intent intent = new Intent(this, LaunchOnceActivity.class);
            intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            startActivity(intent);
        }
    }
}

MainActivity.java:

package tests.globalactivitytest;

import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
}

LaunchOnceActivity.java:

package tests.globalactivitytest;

import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;

public class LaunchOnceActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_launch_once);
    }
}

清单文件:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="tests.globalactivitytest"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="23" />

    <application
        android:name=".activity.GlobalActivity"
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity
            android:name=".LaunchOnceActivity"
            android:label="@string/title_activity_launch_once" >
        </activity>
    </application>

</manifest>

【问题讨论】:

  • 如果没有某种“闪存”,您将无法真正处理这个问题。我的建议是,将启动屏幕视图作为您启动的活动并从那里处理重定向,这样无论加载哪个活动,您都可以获得持续的体验。
  • 您应该使用某种 SplashScreen(如@zgc7009 所说)以及使用应用程序版本等(共享首选项将是存储数据的好地方)。官方 Android 不提供您打算做的事情,这是缺点。
  • @zgc7009 你看过 Instagram 吗? Instagram、Quora、WhatsApp,它们都要求您在首次启动应用程序时注册或登录。他们使用这样的黑客吗?我不认为他们会使用 hacky 解决方案,因为他们是科技巨头并且拥有最好的工程师。 :s
  • 是的,当然。这根本不是 hack,它实际上是 Android 开发的一个非常常见的部分,我的很多应用程序都有启动画面。它们甚至记录在官方文档中。史密斯先生在下面有一个可行的答案。
  • 尝试使用可见性。

标签: android android-activity oncreate onresume android-launcher


【解决方案1】:

上面的答案也很好,但如果你需要在没有飞溅活动的情况下这样做,那么你可以使用下面的方法

第 1 步:创建一个新类并从 Application 扩展并添加以下代码(不要忘记相应地更改您的活动名称)。其中最重要的部分是在开始任何活动之前设置适当的 Intent Flags。

此 GlobalActivity 将在您的启动器 Activity 之前调用

public class GlobalActivity extends Application {


@Override
public void onCreate() {
    super.onCreate();
  SharedPreferences userInfo = PreferenceManager.getDefaultSharedPreferences(getApplicationContext());
  SharedPreferences.Editor  editor = userInfo.edit();
  boolean logedIn =  userInfo.getBoolean("loggedIn", false);
    if (logedIn)
    {
        Intent intent = new Intent(this,MainActivity.class);
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
        startActivity(intent);
    }
    else {
        Intent intent = new Intent(this,LaunchOnceActivity.class);
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK);
        startActivity(intent);
    }
}
}

第 2 步: 您的清单文件应如下所示

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


<application
android:name=".GlobalActivity"
android:allowBackup="true"
android:icon="@drawable/launch"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
    android:name=".MainActivity"
    android:label="@string/app_name" >
    <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
</activity>
<activity
    android:name=".LaunchOnceActivity"
    android:label="Launch Once" >
</activity>
</application>
</manifest>

然后相应地更改任何活动中的loggedIn变量以供将来使用。

【讨论】:

  • 是否有理由在onCreate() 的第二个语句中使用getApplicationContext() 而不是this?因为您在需要Context 的其他地方(即在startActivity() 语句中)使用了this
  • 我应该在哪里添加启动器意图过滤器?
  • launcher intent filter 和往常一样,没关系,只需要先添加你想启动的activity,上面GlobalActivity的onCreate方法会在Launcher activity的onCreate之前调用。跨度>
  • 如果你想在 GlobalActivity 中决定启动器,你可以像上面一样。只需将任何活动设置为启动器。
  • getApplicationContext() 在这里都是一样的,因为你是从应用程序扩展而来的。
【解决方案2】:

您可以创建一个启动活动,并从其 onCreate 方法检查接下来要运行的活动,等待几秒钟,启动该活动,然后立即完成启动。

如果你不想创建splash,那么创建一个没有GUI内容和半透明主题的假splash activity,并在onCreate中进行路由,然后立即完成。这将非常快,用户不会注意到。

【讨论】:

  • 我喜欢这个响应,它提供了一个选项,不会强制显示初始屏幕,但仍会隐藏 blip/flash。如果您有时间发布一个示例,那就太棒了,但仍然是一个很好的建议。
  • 你看过 Instagram 吗? Instagram、Quora、WhatsApp,它们都要求您在首次启动应用程序时注册或登录。他们使用这样的黑客吗?我认为他们不会使用骇人听闻的解决方案,因为他们是科技巨头。 :s
  • @Solace 这可以处理这种情况。
  • @Solace 好吧,如果您设法找到这些公司是如何做到的,请在此处发布,但恐怕它不是开源代码 :) 仍然对他们的方法感到好奇。
  • @Solace 还注意到 Instagram、WhatsApp、Facebook 和其他应用程序可能没有最佳或最用户友好的做法。以 Facebook 为例。该应用使用的资源相当于整个 Android 操作系统的 20%(在 Nexus 5X 上测试)。我不一定会尝试追随他们的脚步。
【解决方案3】:

您可以将您的活动和正常启动器包装在片段中。然后,在其 onCreate 中使用片段管理器创建另一个活动,并始终将其设为您的启动器。在该活动中将正确的片段放在屏幕上之前检查首选项。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2023-03-23
    • 1970-01-01
    • 2013-12-23
    • 2012-06-10
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多