android Activity笔记

安卓四大组件之一

创建Activity

  • 定义java类,继承Activity
  • 在清单文件中配置activity标签

多个入口

  • 在桌面有多个应用,一个应用对应一个入口
  • 一个应用程序可以在桌面创建多个快捷图标。
  • activity的名称、图标可以和应用程序的名称、图标不相同
<application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="com.xxx.createactivity.MainActivity"
            android:label="Main" 
            android:icon="@drawable/photo2"
            >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity android:icon="@drawable/photo3"
                 android:label="Second"
                 android:name="com.xxx.createactivity.SecondActivity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
        </activity>
    </application>

跳转

显式跳转
  • 同一应用中,指定目标Activity的字节码和当前的上下文
  • 不同应用中,指定目标Activity所在的应用的包名和目标Activity的包名加类名
Intent intent = new Intent();
intent.setClass(this, SecondActivity.class);
startActivity(intent);

Intent intent = new Intent();
//启动系统自带的拨号器应用
intent.setClassName("com.android.dialer", "com.android.dialer.DialtactsActivity");
startActivity(intent);
隐式跳转
  • 系统会在所有清单文件中寻找与程序员创建的intent匹配的intent-filter,找到则启动,找不到则抛异常
  • 匹配就是intent-filter中定义了什么属性,程序员创建的intent中也必须设置什么属性
  • 在清单文件的activity标签下,配置intent-filter子节点,其中再配置action和category
    • action 指定动作(可以自定义,可以使用系统自带的)
    • data 指定数据(操作什么内容)
    • category 类别 (默认类别,机顶盒,车载电脑)
    • 隐式意图启动Activity,需要为intent设置以上三个属性,且值必须与该Activity在清单文件中对三个属性的定义匹配
    • intent-filter节点及其子节点都可以同时定义多个,隐式启动时只需与任意一个匹配即可
<intent-filter >
    <action android:name="com.xxx.second"/>//必须有,可以有多个(一般只有一个),匹配一个即可
    <data android:scheme="asd" android:mimeType="text/name"/>//可有可无,可以有多个,匹配一个即可,传递不同的数据
    <category android:name="android.intent.category.DEFAULT"/>//必须有
</intent-filter>
public void click5(View v){
    Intent intent = new Intent();
    intent.setAction("com.xxx.second");
    //匹配mimetype
//      intent.setType("text/name"); //会清除data
    //匹配scheme
//      intent.setData(Uri.parse("asd:heooooo")); //会清除mimetype
    intent.setDataAndType(Uri.parse("asd:heooooo"), "text/name");
    //如果没有设置Category的匹配,那么自动匹配CATEGORY_DEFAULT
//      intent.addCategory(Intent.CATEGORY_DEFAULT);
    startActivity(intent);
}

//获取通过setData传递的数据
//获取启动此Activity的intent对象
Intent intent = getIntent();
Uri uri = intent.getData();
应用场景
  • 启动同一应用中的Activity,用显式
  • 启动不同应用中的Activity,用隐式
  • 显式启动效率高于隐式
  • 如果系统找到了多个intent-filter与程序员创建的intent匹配,那么就会弹出对话框,列举所有匹配的Activity,让用户选择

数据传递

  • Activity跳转时,可以把数据封装在intent对象中
  • intent中可以封装的数据类型:八大基本数据类型和字符串及它们的数组,还有实现了序列化接口的对象,还有bundle对象
  • 数据可以先封装至Bundle,再把Bundle封装至intent
//Activity通过Intent启动时,可以通过Intent对象携带数据到目标Activity
Intent intent = new Intent(this, SecondActivity.class);
//intent.putExtra("maleName", maleName);
//intent.putExtra("femaleName", femaleName);

//Bundle中也可以用键值对封装数据,封装类型与intent一致
Bundle extras = new Bundle();
extras.putString("maleName", maleName);
extras.putString("femaleName", femaleName);
intent.putExtras(extras);
startActivity(intent);

//在目标Activity中取出数据
Intent intent = getIntent();
//String maleName = intent.getStringExtra("maleName");
//String femaleName = intent.getStringExtra("femaleName");

Bundle bundle = intent.getExtras();
String maleName = bundle.getString("maleName");
String femaleName = bundle.getString("femaleName");

横竖屏切换

  • 默认情况横竖屏切换会触发生命周期方法重新执行,Activity销毁重建
  • 用以下代码让横竖屏切换时不重建Activity

    android:configChanges="orientation|screenSize|keyboardHidden"
    
  • 用以下代码写死屏幕方向

    android:screenOrientation="portrait"
    

生命周期

void onCreate()
  • Activity已经被创建完毕
void onStart()
  • Activity已经显示在屏幕,但没有得到焦点
void onResume()
  • Activity得到焦点,可以与用户交互
void onPause()
  • Activity失去焦点,无法再与用户交互,但依然可见
void onStop()
  • Activity不可见,进入后台
void onDestroy()
  • Activity被销毁, 按返回键会执行到这,但进程还在
void onRestart()
  • Activity从不可见变成可见时会执行此方法

使用场景

  • Activity创建时需要初始化资源,销毁时需要释放资源;或者播放器应用,在界面进入后台时需要自动暂停

完整生命周期(entire lifetime)

onCreate–>onStart–>onResume–>onPause–>onStop–>onDestory

可视生命周期(visible lifetime)

onStart–>onResume–>onPause–>onStop

前台生命周期(foreground lifetime)

onResume–>onPause

四种启动模式

每个应用会有一个Activity任务栈,存放已启动的Activity

Activity的启动模式,修改任务栈的排列情况

  • standard 标准启动模式
  • singleTop 单一顶部模式
    • 如果任务栈的栈顶存在这个要开启的activity,不会重新的创建activity,而是复用已经存在的activity。保证栈顶如果存在,不会重复创建。
    • 应用场景:浏览器的书签
  • singeTask 单一任务栈,在当前任务栈里面只能有一个实例存在

    • 当开启activity的时候,就去检查在任务栈里面是否有实例已经存在,如果有实例存在就复用这个已经存在的activity,并且把这个activity上面的所有的别的activity都清空,复用这个已经存在的activity。保证整个任务栈里面只有一个实例存在
    • 应用场景:浏览器的activity
    • 如果一个activity的创建需要占用大量的系统资源(cpu,内存)一般配置这个activity为singletask的启动模式。webkit内核 c代码
  • singleInstance启动模式非常特殊, activity会运行在自己的任务栈里面,并且这个任务栈里面只有一个实例存在(跳转到其他activity时 切换到后台,切换回来时 显示在前台)

    • 如果你要保证一个activity在整个手机操作系统里面只有一个实例存在,使用singleInstance
    • 应用场景: 电话拨打界面

横竖屏切换的生命周期

默认情况下 ,横竖屏切换, 销毁当前的activity,重新创建一个新的activity

快捷键ctrl+F11

在一些特殊的应用程序常见下,比如游戏,不希望横竖屏切换activity被销毁重新创建
需求:禁用掉横竖屏切换的生命周期

  1. 横竖屏写死
    android:screenOrientation=”landscape”
    android:screenOrientation=”portrait”

  2. 让系统的环境 不再去敏感横竖屏的切换。

     android:configChanges="orientation|screenSize|keyboardHidden"
    

开启activity并获取返回值

从A界面打开B界面, B界面关闭的时候,返回一个数据给A界面

步骤:
  1. 开启activity并且获取返回值

    startActivityForResult(intent, 0);
    
  2. 在新开启的界面里面实现设置数据的逻辑

    Intent data = new Intent();
    data.putExtra("phone", phone);
    //设置一个结果数据,数据会返回给调用者
    setResult(0, data);
    finish();//关闭掉当前的activity,才会返回数据
    
  3. 在开启者activity里面实现方法

    //通过data获取返回的数据
    onActivityResult(int requestCode, int resultCode, Intent data) {
    
    }
    
  4. 通过判断请求码和结果码确定返回值的作用

    public class MainActivity extends Activity {

    public static final int CONTACT = 10;
    public static final int SMS = 20;
    public static final int HANDLE_EXCEPTION = 30;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }


    public void click(View v){
        Intent intent = new Intent(this, ContactActivity.class);
        //启动一个Activity去获取数据(结果)
        startActivityForResult(intent, CONTACT);
    }
    public void click2(View v){
        Intent intent = new Intent(this, SmsActivity.class);
        startActivityForResult(intent, SMS);
    }
    public void click3(View v){
        Intent intent = new Intent(this, HandleExceptionActivity.class);
        startActivityForResult(intent, HANDLE_EXCEPTION);
    }

    //只有通过startActivityForResult启动的Activity销毁时,才会回调这个方法,方法中传入的intent就封装了返回的数据
    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);

        //通过请求码来判断data来自哪一个Activity
        if(requestCode == CONTACT){
            String name = data.getStringExtra("name");

            EditText et_name = (EditText) findViewById(R.id.et_name);
            et_name.setText(name);
        }
        else if(requestCode == SMS){
            String sms = data.getStringExtra("sms");
            EditText et_content = (EditText) findViewById(R.id.et_content);
            et_content.setText(sms);
        }
        //先通过请求码判断data来自哪个Activity
        else if(requestCode == HANDLE_EXCEPTION){
            //通过结果码判断数据data中的数据是什么类型
            if(resultCode == 100){
                String name = data.getStringExtra("name");
                EditText et_name = (EditText) findViewById(R.id.et_name);
                et_name.setText(name);
            }
            else if(resultCode == 200){
                String sms = data.getStringExtra("sms");
                EditText et_content = (EditText) findViewById(R.id.et_content);
                et_content.setText(sms);
            }
        }
    }

}

public class ContactActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        setContentView(R.layout.activity_contact);
        final String[] names = new String[]{
                "春晓",
                "苗润",
                "爱的结晶"
        };
        //显示联系人列表
        ListView lv = (ListView) findViewById(R.id.lv);
        lv.setAdapter(new ArrayAdapter<String>(this, R.layout.item_listview, R.id.tv, names));

        lv.setOnItemClickListener(new OnItemClickListener() {
            //position:点击某个条目时,会通过position告诉程序员点击的是哪一个条目
            @Override
            public void onItemClick(AdapterView<?> parent, View view,
                    int position, long id) {

                Intent data = new Intent();
                //把要传递的数据封装至intent中
                data.putExtra("name", names[position]);
                //当此Activity销毁时,返回至上一个Activity时,会把这个intent对象传递给上一个Activity
                setResult(10, data);

                //销毁当前Activity
                finish();
            }

        });
    }
}

public class HandleExceptionActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_handleexception);

        final TextView tv_name = (TextView) findViewById(R.id.tv_name);
        final TextView tv_content = (TextView) findViewById(R.id.tv_content);

        tv_name.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                Intent data = new Intent();
                data.putExtra("name", tv_name.getText());

                setResult(100, data);
                finish();
            }
        });

        tv_content.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View v) {
                Intent data = new Intent();
                data.putExtra("sms", tv_content.getText());

                setResult(200, data);
                finish();
            }
        });
    }
}

public class SmsActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_contact);

        final String[] smss = new String[]{
                "我正在开会,一会回你",
                "我正在7饭,一会回你",
                "我已死,有事烧纸"
        };
        //显示联系人列表
        ListView lv = (ListView) findViewById(R.id.lv);
        lv.setAdapter(new ArrayAdapter<String>(this, R.layout.item_listview, R.id.tv, smss));

        lv.setOnItemClickListener(new OnItemClickListener() {

            @Override
            public void onItemClick(AdapterView<?> parent, View view,
                    int position, long id) {
                Intent data = new Intent();
                data.putExtra("sms", smss[position]);

                setResult(20, data);
                finish();

            }
        });
    }
}

finish()只是通知要销毁activity,后面的代码还是会走完的,例如activity切换时动画的代码必须写在finish()后

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值