Home

hogen

Follow Excellence. Success will chase you.

Home About Github Email

Android 生命周期及 Intent 使用

此博内容为读第二行代码时所记的笔记总结

返回栈

Android 是使用任务栈(Task)来管理活动的,一个任务就是一组存放在栈里的活动的集合,这个栈也被称作返回栈(Back Stack),在默认情况下,每当我们启动了一个新的活动,它会在返回栈中入栈,并处于栈顶的位置。而每当我们
按下 Back键或调用 finish() 方法去销毁一个活动时,处于栈顶的活动会出栈,这时前一个入
栈的活动就会重新处于栈顶的位置。系统总是会显示处于栈顶的活动给用户

活动周期

  • onCreate(): 在这个方法中完成活动的初始化操作,比如说加载布局、绑定事件等
  • onStart(): 这个方法在活动由不可见变为可见的时候调用。
  • onResume():这个方法在活动准备好和用户进行交互的时候调用。 此时的活动一定位于返回栈的
    栈顶,并且处于运行状态。
  • onPause(): 这个方法在系统准备去启动或者恢复另一个活动的时候调用。 我们通常会在这个方
    法中将一些消耗 CPU 的资源释放掉,以及保存一些关键数据,但这个方法的执行速度
    一定要快,不然会影响到新的栈顶活动的使用。
  • onStop(): 这个方法在活动完全不可见的时候调用。它和 onPause()方法的主要区别在于,如
    果启动的新活动是一个对话框式的活动,那么 onPause() 方法会得到执行,而 onStop() 方法并不会执行。
  • onDestory(): 这个方法在活动被销毁之前调用,之后活动的状态将变为销毁状态。
  • onRestart(): 这个方法在活动由停止状态变为运行状态之前调用,也就是活动被重新启动了

Android活动的四种启动模式

  • standard: standard 是活动默认的启动模式,在不进行显式指定的情况下,所有活动都会自动使用这种启动模式。在 standard 模式(即默认情况)下,每当启动一个新的活动,它就会在返回栈中入栈,并处于栈顶的位置。对于使用standard模式的活动,系统不会在乎这个活动是否已经在返回栈中存在,每次启动都会创建该活动的一个新的实例。

  • singleTop: 当活动的启动模式指定为 singleTop,在启动活动时如果发现返回栈的栈顶已经是该活动, 则认为可以直接使用

  • singleTask: 当活动的启动模式指定为 singleTask,每次启动该活动时系统首先会在返回栈中检查是否存在该活动的实例,如果发现已经存在则直接使用该实例,并把在这个活动之上的所有活动统统出栈,如果没有发现就会创建一个新的活动实例。不会再创建新的活动实例。

  • singleInstance: 指定为 singleInstance 模式的活动会启用一个新的返回栈来管理这个活动(其实如果 singleTask模式指定了不同的 taskAffinity,也会启动一个新的返回栈)。想象以下场景,假设我们的程序中有一个活动是允许其他程序调用的, 如果我们想实现其他程序和我们的程序可以共享这个活动的实例,应该如何实现呢?使用前面三种启动模式肯定是做不到的,因为每个应用程序都会有自己的返回栈,同一个活动在不同的返回栈中入栈时必然是创建了新的实例。而使用singleInstance模式就可以解决这个问题, 在这种模式下会有一个单独的返回栈来管理这个活动,不管是哪个应用程序来访问这个活动,都共用的同一个返回栈,也就解决了共享活动实例的问题。

使用显示Intent

点击FirstActivity的Button按钮,核心代码如下

button1.setOnClickListener(new OnClickListener() {
    @Override
    public void onClick(View v) {
        Intent intent = new Intent(FirstActivity.this, SecondActivity.class);//"意图很明显,从FirstActivity到SecondActivity"
        startActivity(intent);
    }
});

并在清单文件AndroidManifest.xml对SecondActivity注册

<application
    android:allowBackup="true"
    android:icon="@drawable/ic_launcher"
    android:label="@string/app_name"
    android:theme="@style/AppTheme" >
    <activity
        android:name=".FirstActivity"
        android:label="This is FirstActivity" >
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter>
    </activity>
    <activity android:name=".SecondActivity" >
    </activity>
</application>

隐式Intent

相比于显式 Intent,隐式 Intent 并不明确指出我们想要启动哪一个活动,而是指定了一系列更为抽象的 action和 category 等信息, 然后交由系统去分析这个 Intent,并帮我们找出合适的活动去启动。

通过在<activity>标签下配置<intent-filter>的内容,可以指定当前活动能够响应的 action
和 category,重新配置 AndroidManifest.xml,添加如下代码:

<activity android:name=".SecondActivity" >
    <intent-filter>
        <action android:name="com.excel.activitytest.ACTION_START" />
        <category android:name="android.intent.category.DEFAULT" />
    </intent-filter>
</activity>

修改 FirstActivity 中按钮的点击事件,代码如下所示:

button1.setOnClickListener(new OnClickListener() {
    @Override
    public void onClick(View v) {
        //表明我们想要启动能够响应 com.excel.activitytest.ACTION_START这个 action的活动
        Intent intent = new Intent("com.excel.activitytest.ACTION_START");
        startActivity(intent);
    }
});

使用Inent传值总结

  • 常规方式,在FirstActivity单击按钮时用putExtra把要传的值放入Intent中,

    public void onclick(View view) {
        Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
        intent.putExtra("data", "Liu.Heguang");//通过键值对的形式
        startActivity(intent);
    }
    

    在SecondActivity通过getStringExtra方法取值

    public class SecondActivity extends Activity {
        private TextView tv_content;
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            // TODO Auto-generated method stub
            super.onCreate(savedInstanceState);
            setContentView(R.layout.second_layout);
            tv_content = (TextView) findViewById(R.id.tv_content);
            String data = getIntent().getStringExtra("data");//通过键名取值
            tv_content.setText(data);
        }
    }
    
  • 有时可以通过Bundle一次性传多个值

    在FirstActivity中传值

    public void onclick(View view) {
        Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
        Bundle bundle = new Bundle();//此处的bundle类此于java中的map集合
        bundle.putInt("age", 22);//把年龄存入bundle
        bundle.putString("name", "张三");//把姓名存入bundle中
        intent.putExtra("bundle", bundle);//把bundle放入intent意图
        startActivity(intent);
    }
    

    在SecondActivity取值

    public class SecondActivity extends Activity {
    private TextView tv_content;//定义一个TextView 对象
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            // TODO Auto-generated method stub
            super.onCreate(savedInstanceState);
            setContentView(R.layout.second_layout);
            tv_content = (TextView) findViewById(R.id.tv_content);
            Bundle bundle = getIntent().getBundleExtra("bundle");//拿到bundle对象
            int age = bundle.getInt("age");//取值
            String name = bundle.getString("name");
            tv_content.setText("年龄:"+age+" 姓名:"+name);//把取得的值显示在TextView中
        }
    }
    

    效果如下:

  • 有时还可以传一个javaBean 对象,前提是这个bean要实现Serializable接口
    定义一个UserBean类如下:

    public class UserBean implements Serializable{
        private static final long serialVersionUID = 1L;
        private int age;
        private String name;
        public UserBean(int age, String name) {
            super();
            this.age = age;
            this.name = name;
        }
    
        @Override
        public String toString() {
            return "UserBean [age=" + age + ", name=" + name + "]";
        }
    }
    

    在FirstActivity中传值

    public void onclick(View view) {
        Intent intent = new Intent(FirstActivity.this, SecondActivity.class);
        UserBean userBean = new UserBean(22, "李四");
        intent.putExtra("userBean", userBean);
        startActivity(intent);
    }
    

    在SecondActivity取值

    public class SecondActivity extends Activity {
        private TextView tv_content;
            @Override
            protected void onCreate(Bundle savedInstanceState) {
                // TODO Auto-generated method stub
                super.onCreate(savedInstanceState);
                setContentView(R.layout.second_layout);
                tv_content = (TextView) findViewById(R.id.tv_content);
                UserBean userBean = (UserBean) getIntent().getSerializableExtra("userBean");
                tv_content.setText(userBean.toString());
            }
    }
    

    效果如下图:


hogen

2017-05-10
Home About Github Email