zoukankan      html  css  js  c++  java
  • Android笔记(五) Activity的启动模式

             Android中Activity是由返回栈来管理的,在默认情况下,每当启动一个新的Activity,它都会在返回栈中入栈,并且出于栈的顶端。但是有些时候Activity已经在栈的顶端了,也就不需要再启动的时候重新创建一个Activity的实例了,所以我们就需要其他的启动方式。

             Activity的启动方式一共分为四种:standard、singleTop、singleTask、singleInstance,可以在AndroidManiFest.xml中通过<activity>标签指定android:launchMode属性来选择启动模式。

             standard模式:该模式为Activity默认的启动模式,如果AndroidManiFest.xml中没有进行配置,系统默认使用这种方式启动,也就是每启动一个Activity,都会在返回栈的顶端插入一个新的Activity实例,我们代码来查看效果

             1)新建LaunchModeDemo工程,主Activity命名为FirstActivity.java

             2)FirstActivity.java

    package cn.lixyz.laubchmode;
    
    import android.content.Intent;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.View;
    
    public class FirstActivity extends AppCompatActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_first);
    
            Log.d("FirstActivity","运行了onCreate()方法创建了一个Activity实例");
    
            findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    //从FirstActivity跳转到FirstActivity
                    Intent intent = new Intent(FirstActivity.this,FirstActivity.class);
                    startActivity(intent);
                }
            });
    
        }
    }

             3)activity_first.xml

    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
        android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
        android:paddingRight="@dimen/activity_horizontal_margin"
        android:paddingTop="@dimen/activity_vertical_margin"
        android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".FirstActivity">
    
    
        <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="点我跳转到另一个Activity"
            android:id="@+id/button"
            android:layout_marginTop="53dp"
            android:layout_alignParentTop="true"
            android:layout_centerHorizontal="true" />
    </RelativeLayout>

             4)AndroidManiFest.xml

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        package="cn.lixyz.laubchmode" >
    
        <application
            android:allowBackup="true"
            android:icon="@mipmap/ic_launcher"
            android:label="@string/app_name"
            android:theme="@style/AppTheme" >
            <activity
                android:name=".FirstActivity"
                android:label="@string/app_name" >
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
    
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
        </application>
    
    </manifest>

      运行结果,没点击一次按钮,都会运行一次onCreate()方法创建一个Activity的实例

      

      如果连续点击三次按钮,我们需要点击四次back才能退出到桌面,由此可见,每次点击按钮,我们都在返回栈中新创建了一个Activity放到了返回栈的顶部,我们点击back只是把最上面的Acticity退出了。

      

             singleTop模式:我们会发现,有时候明明Activity已经在栈顶端了,再创建一个实例放到顶端,这不是浪费资源么,这时候我们就需要对Activity的启动模式进行改动了,如果Activity设置为singleTop模式,那么在启动时候发现栈顶已经是该Activity的时候,系统不会在创建新的Activity实例了。

             通过代码来看:

             1)FirstActivity.java

    package cn.lixyz.laubchmode;
    
    import android.content.Intent;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.View;
    
    public class FirstActivity extends AppCompatActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_first);
    
            Log.d("FirstActivity","运行了onCreate()方法创建了一个Activity实例");
    
            findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    //从FirstActivity跳转到FirstActivity
                    Intent intent = new Intent(FirstActivity.this,FirstActivity.class);
                    startActivity(intent);
                }
            });
    
        }
    }

             2)activity_first.xml

    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
        android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
        android:paddingRight="@dimen/activity_horizontal_margin"
        android:paddingTop="@dimen/activity_vertical_margin"
        android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".FirstActivity">
    
    
        <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="点我跳转到另一个Activity"
            android:id="@+id/button"
            android:layout_marginTop="53dp"
            android:layout_alignParentTop="true"
            android:layout_centerHorizontal="true" />
    </RelativeLayout>

             3)AndroidManiFest.xml

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        package="cn.lixyz.laubchmode" >
    
        <application
            android:allowBackup="true"
            android:icon="@mipmap/ic_launcher"
            android:label="@string/app_name"
            android:theme="@style/AppTheme" >
            <activity
                android:name=".FirstActivity"
                android:label="@string/app_name"
                android:launchMode="singleTop">
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
    
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
        </application>
    </manifest>

      运行结果,只有我们启动的时候运行了一次onCreate()方法,后边点击按钮则不会再创建新的Activity实例

             singleTask模式:我们使用singleTop可以使得Activity在返回栈顶端则不创建新的Activity实例,但如果Activity不在顶端,依然还会创建,如何让Activity在整个应用程序的上下文中只存在一个实例呢?这就要借助sigleTask来实现了

             当一个Activity设定启动模式为singleTask的话,每次启动该活动时系统会先在整个返回栈中检查一遍,如果返回栈中已经存在该Activity的实例的话,则会直接使用该实例,并且把这个Activity之上的Activity统统出栈,如果返回栈中不存在该Activity实例,则创建一个。

             通过代码来看:

             1)FirstActivity.java

    package cn.lixyz.laubchmode;
    
    import android.content.Intent;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.View;
    
    public class FirstActivity extends AppCompatActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_first);
    
            Log.d("Launch","运行了FirstActivity的onCreate()方法");
    
            findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    //从FirstActivity跳转到FirstActivity
                    Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
                    startActivity(intent);
                }
            });
    
        }
    
        @Override
        protected void onRestart() {
            super.onRestart();
            Log.d("Launch","运行了FirstActivity的onRestart()方法");
        }
    }

             2)activity_first.xml

    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
        android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
        android:paddingRight="@dimen/activity_horizontal_margin"
        android:paddingTop="@dimen/activity_vertical_margin"
        android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".FirstActivity">
    
    
        <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="点我跳转到另一个Activity"
            android:id="@+id/button"
            android:layout_marginTop="53dp"
            android:layout_alignParentTop="true"
            android:layout_centerHorizontal="true" />
    </RelativeLayout>

             3)SecondActivity.java

    package cn.lixyz.laubchmode;
    
    import android.content.Intent;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.Menu;
    import android.view.MenuItem;
    import android.view.View;
    
    public class SecondActivity extends AppCompatActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_second);
            Log.d("Launch", "运行了SecondActivity的onCreate方法");
            findViewById(R.id.button2).setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    Intent intent = new Intent(SecondActivity.this,FirstActivity.class);
                    startActivity(intent);
                }
            });
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            Log.d("Launch", "运行了SecondActivity的onDestroy方法");
        }
    }

             4)activity_second.xml

    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
        android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
        android:paddingRight="@dimen/activity_horizontal_margin"
        android:paddingTop="@dimen/activity_vertical_margin"
        android:paddingBottom="@dimen/activity_vertical_margin"
        tools:context="cn.lixyz.laubchmode.SecondActivity">
    
    
        <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="点我跳转"
            android:id="@+id/button2"
            android:layout_alignParentTop="true"
            android:layout_centerHorizontal="true"
            android:layout_marginTop="94dp" />
    </RelativeLayout>

             5)AndroidManiFest.xml

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        package="cn.lixyz.laubchmode" >
    
        <application
            android:allowBackup="true"
            android:icon="@mipmap/ic_launcher"
            android:label="@string/app_name"
            android:theme="@style/AppTheme" >
            <activity
                android:name=".FirstActivity"
                android:label="@string/app_name"
                 android:launchMode="singleTask">
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
    
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
            <activity
                android:name=".SecondActivity"
                android:launchMode="singleTask">
            </activity>
        </application>
    </manifest>

      运行结果:

      程序启动,检查返回栈中没有FirstActivity的实例,调用FirstActivity的onCreate()方法创建一个实例,点击按钮跳转到SecondActivity,发现移动栈中没有SecondActivyt的实例,调用SecondActivity的onCreate()方法创建一个SecondActivity的实例,点击SecondActivity上的按钮跳转到FirstActivity上,系统发现移动栈中有FirstActivity的实例,调用FirstActivity的onRestart()方法,并且将FirstActivity实例上面的所有实例全部移除(调用SecondActivity的onDestroy()方法),再次点击FirstActivity上的按钮跳转到SecondActivity上,系统发现移动栈中没有SecondActivity的实例,于是继续调用它的onCreate()方法创建之。

             singleInstance模式:指定为singleInstance模式的Activity会启用一个新的返回栈来管理这个活动。

             假如我们的程序中有一个Activity是运行其他程序调用的,如果我们想实现其他程序和我们的程序可以共享这个Activity实例,上面三个启动模式是无法做到的,因为每个程序都会有自己的返回栈,而使用singleInstance模式的话,有一个单独的返回栈来管理这个Activity,不论那个程序访问这个Activity,都共用同一个返回栈。

             通过代码来看:

             1)FirstActivity.java

    package cn.lixyz.laubchmode;
    
    import android.content.Intent;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.View;
    
    public class FirstActivity extends AppCompatActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_first);
    
            Log.d("Launch","运行了FirstActivity的onCreate()方法,TaskId= " + getTaskId());
    
            findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    //从FirstActivity跳转到FirstActivity
                    Intent intent = new Intent(FirstActivity.this,SecondActivity.class);
                    startActivity(intent);
                }
            });
    
        }
    
        @Override
        protected void onRestart() {
            super.onRestart();
            Log.d("Launch","运行了FirstActivity的onRestart()方法,TaskId= " + getTaskId()
            );
        }
    }

             2)activity_first.xml

    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
        android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
        android:paddingRight="@dimen/activity_horizontal_margin"
        android:paddingTop="@dimen/activity_vertical_margin"
        android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".FirstActivity">
    
    
        <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="点我跳转到另一个Activity"
            android:id="@+id/button"
            android:layout_marginTop="53dp"
            android:layout_alignParentTop="true"
            android:layout_centerHorizontal="true" />
    </RelativeLayout>

             3)SecondActivity.java

    package cn.lixyz.laubchmode;
    
    import android.content.Intent;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.Menu;
    import android.view.MenuItem;
    import android.view.View;
    
    public class SecondActivity extends AppCompatActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_second);
            Log.d("Launch", "运行了SecondActivity的onCreate方法,TaskId= " + getTaskId());
            findViewById(R.id.button2).setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    Intent intent = new Intent(SecondActivity.this,FirstActivity.class);
                    startActivity(intent);
                }
            });
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            Log.d("Launch", "运行了SecondActivity的onDestroy方法,TaskId= " + getTaskId());
        }
    }

             4)activity_second.xml

    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
        android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
        android:paddingRight="@dimen/activity_horizontal_margin"
        android:paddingTop="@dimen/activity_vertical_margin"
        android:paddingBottom="@dimen/activity_vertical_margin"
        tools:context="cn.lixyz.laubchmode.SecondActivity">
    
    
        <Button
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="点我跳转"
            android:id="@+id/button2"
            android:layout_alignParentTop="true"
            android:layout_centerHorizontal="true"
            android:layout_marginTop="94dp" />
    </RelativeLayout>

             5)AndroidManiFest.xml

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        package="cn.lixyz.laubchmode" >
    
        <application
            android:allowBackup="true"
            android:icon="@mipmap/ic_launcher"
            android:label="@string/app_name"
            android:theme="@style/AppTheme" >
            <activity
                android:name=".FirstActivity"
                android:label="@string/app_name"
                 android:launchMode="singleTask">
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
    
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
            <activity
                android:name=".SecondActivity"
                android:launchMode="singleInstance">
            </activity>
        </application>
    </manifest>

      运行结果:

      发现跳转到SecondActivity的时候,TaskId变了,因为SecondActivity的启动方式是singleInstance,它启动了一个新的返回栈来管理这个Activity

  • 相关阅读:
    机器学习、图像识别方面 书籍推荐 via zhihu
    网络工具 NetCat
    CSharp读取配置文件的类(简单实现)
    about future
    Google's BBR拥塞控制算法模型解析
    对称加密与非对称加密
    windows平台下新网络库RIO ( Winsock high-speed networking Registered I/O)
    在mac os下编译android -相关文章
    [原创] linux 下上传 datapoint数据到yeelink 【golang版本】同时上传2个数据点
    在 树莓派上使用 c++ libsockets library
  • 原文地址:https://www.cnblogs.com/xs104/p/4712386.html
Copyright © 2011-2022 走看看