★04.Activity

项目里添加Activity

  • Activity对应的 Java 文件
  • AndroidManifest.xml
  • 对应的界面 XML
  • 界面 XML 用到的资源

Activity及其派生类

继承图

简介

简单示例

AndroidManifest.xml

<activity
        android:name=".MainActivity"
        android:launchMode="singleTop"
        android:screenOrientation="portrait">
    <intent-filter>
        <action android:name="android.intent.action.MAIN"/>
        <category android:name="android.intent.category.LAUNCHER"/>
    </intent-filter>
</activity>

MainActivity.java

public class MainActivity extends AppCompatActivity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
}

启动模式

  • standard
    • 默认模式,可以不用写配置。
    • 在原任务栈中创建新Activity,不会创建新栈。
    • 可以有多个相同或不同的Activity
  • singleTop
    • 在原任务栈中创建新Activity,不会创建新栈。
    • 可以有多个相同Activity
    • 如果Activity在任务栈顶的时候,启动与栈顶相同的Activity时,不会创建新的Activity,而会调用栈顶ActivityonNewIntent()
  • singleTask
    • 在原任务栈中创建新Activity,不会创建新栈。
    • 启动新Activity时,若任务栈中已经存在相同的Activity,就会销毁任务栈中这个Activity之上的所有Activity,并调用该ActivityonNewIntent()
  • singleInstance
    • 在新的任务栈中创建新的Activity
    • 在新任务栈中只能有这一个相同或不同的Activity

常见技巧

  • onCreate()中检查savedInstanceState是否为null,若是则代表着Activity是第一次加载,同样的技巧在Fragment中也是可用的。
  • 通过getIntent()来获取调用时传入的数据。

常用方法

通过重写使用

  • onActivityReenter():点击返回键时,重新进入父Activity时调用,会接收到在子Activity中调用setResult()时传送的数据。
  • onActivityResult():在子Activity中调用setResult(),然后退出子Activity时,父Activity会接收到在setResult()中设置的数据。
  • finishAfterTransition():在退场动画结束后执行finish(),通过重写来决定在退场动画结束与调用finish()之间需要做的东西。

直接调用

  • postponeEnterTransition()
    • 当通过ActivityOptions.makeSceneTransitionAnimation()启动Activity时,延迟开始转场动画,直至调用startPostponedEnterTransition()
    • 若没有通过ActivityOptions.makeSceneTransitionAnimation()启动Activity,则此函数什么都不做。
    • onCreate()onActivityReenter()中调用。
  • startPostponedEnterTransition():开始转场动画。

生命周期

  • 打开
    • onCreate()
    • onStart()
    • onResume()
  • 进入后台
    • onPause()
    • onStop()
  • 进入前台
    • onRestart()
    • onStart()
    • onResume()
  • 应用程序配置发生改变(横竖屏切换)
    • onPause()
    • onStop()
    • onDestrory()
    • onCreate()
    • onStart()
    • onResume()
  • 退出
    • onPause()
    • onStop()
    • onDestrory()

横竖屏布局

通过创建默认方向layout文件和横向layout文件来实现。

保存Activity数据

  • 只能保存基本数据类型和实现SerializableParcelable接口的对象。
  • 应用配置改变时会调用onSaveInstanceState()
  • 保存:重写onSaveInstanceState()
        @Override
        protected void onSaveInstanceState(Bundle outState) {
            super.onSaveInstanceState(outState);
            outState.putInt(KEY_INDEX, mCurrentIndex);
        }
    
  • 读取:在onCreate()中恢复数据
    if (savedInstanceState != null) {
        mCurrentIndex = savedInstanceState.getInt(KEY_INDEX, 0);
    }
    

启动Activity

准备

  • 被启动的活动必须具有的意图筛选器:
    <category android:name="android.intent.category.DEFAULT"/>
    

方式一

startActivity(new Intent(".MainActivity"));

方式二

startActivity(new Intent(this, MainActivity.class));

Activity通信

传入数据

  • 调用的Activity
    Intent i = new Intent(this, CheatActivity.class);
    i.putExtra(/* data */);
    startActivity(i);
    
  • 被调用的Activity
    mAnswerIsTrue = getIntent().getBooleanExtra(/* data */);
    

返回数据

  1. 启动Activity时,给Activity一个请求标识符:
    startActivityForResult(i, REQUEST_CODE);
    
  2. 在被调用的Activity中设置结果,调用者会接受到此结果:
    Intent data = new Intent();
    data.putExtra(/* data */);
    setResult(RESULT_OK, data);
    
  3. 被调用的Activity结束后会回调 调用者onActivityResult(),数据会通过Intent传回,可以根据请求标识符解析传回的Intent
    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (resultCode != Activity.RESULT_OK) {
            return;
        }
        if (requestCode == REQUEST_CODE) {
            if (data == null) {
                return;
            }
            // 解析数据
            data.getBooleanExtra(/* data */);
        }
    }
    

封装newIntent()静态方法

  • 通过在被调用的Activity中封装newIntent()静态方法,使得被调用的Activity来决定如何被调用:
    public static Intent newIntent(Context packageContext, Data data) {
        Intent intent = new Intent(packageContext, MyActivity.class);
        intent.putExtra(ID, data);
        return intent;
    }
    

注意事项

  • 相比与onResume()onStart()不一定会执行。
  • Activity中的实例变量不可靠,生命期比Activity长的实例变量应该尽量存在外部。
  • 确保所有Activity都是AppCompatActivity的子类,这样兼容性会很高。
  • 后退按钮导航和层级导航(向上按钮导航)并不一样。层级导航是开始一个新的Activity,弹出回退栈里此Activity以上的Activity。后退按钮导航则是弹出回退栈里最顶层的Activity

Activity的XML属性

<activity android:allowEmbedded=["true" | "false"]
          android:allowTaskReparenting=["true" | "false"]
          android:alwaysRetainTaskState=["true" | "false"]
          android:autoRemoveFromRecents=["true" | "false"]
          android:banner="drawable resource"
          android:clearTaskOnLaunch=["true" | "false"]
          android:configChanges=["mcc", "mnc", "locale",
                                 "touchscreen", "keyboard", "keyboardHidden",
                                 "navigation", "screenLayout", "fontScale",
                                 "uiMode", "orientation", "screenSize",
                                 "smallestScreenSize"]
          android:documentLaunchMode=["intoExisting" | "always" |
                                  "none" | "never"]
          android:enabled=["true" | "false"]
          android:excludeFromRecents=["true" | "false"]
          android:exported=["true" | "false"]
          android:finishOnTaskLaunch=["true" | "false"]
          android:hardwareAccelerated=["true" | "false"]
          android:icon="drawable resource"
          android:label="string resource"
          android:launchMode=["standard" | "singleTop" |
                              "singleTask" | "singleInstance"]
          android:maxRecents="integer"
          android:multiprocess=["true" | "false"]
          android:name="string"
          android:noHistory=["true" | "false"]
          android:parentActivityName="string"
          android:permission="string"
          android:process="string"
          android:relinquishTaskIdentity=["true" | "false"]
          android:resizeableActivity=["true" | "false"]
          android:screenOrientation=["unspecified" | "behind" |
                                     "landscape" | "portrait" |
                                     "reverseLandscape" | "reversePortrait" |
                                     "sensorLandscape" | "sensorPortrait" |
                                     "userLandscape" | "userPortrait" |
                                     "sensor" | "fullSensor" | "nosensor" |
                                     "user" | "fullUser" | "locked"]
          android:stateNotNeeded=["true" | "false"]
          android:supportsPictureInPicture=["true" | "false"]
          android:taskAffinity="string"
          android:theme="resource or theme"
          android:uiOptions=["none" | "splitActionBarWhenNarrow"]
          android:windowSoftInputMode=["stateUnspecified",
                                       "stateUnchanged", "stateHidden",
                                       "stateAlwaysHidden", "stateVisible",
                                       "stateAlwaysVisible", "adjustUnspecified",
                                       "adjustResize", "adjustPan"] >
</activity>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 引言:这篇文章,大概分析下Fragment的生命周期、实际应用方法以及使用Fragment时需要注意的地方,算是F...
    androidjp阅读 12,948评论 10 104
  • Fragment文档 简单示例 说明 Activity通过一个FrameLayout来安排Fragment在Act...
    iDragonfly阅读 246评论 0 0
  • 《Android Fragment完全解析,关于碎片你所需知道的一切》 我们都知道,Android上的界面展示都是...
    cxm11阅读 2,205评论 2 19
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 173,067评论 25 708
  • 这是女儿第一次听我当众弹奏古琴。担心她一 个人坐在那里会哭闹,害怕,毕竟只有2岁。古琴跟人心最知心,慌乱的心情暴露...
    文姐28101阅读 269评论 1 2