参考书籍:《第一行代码》 第二版 郭霖
开发工具:AndroidStudio 3.2 Stable Channel
如有错漏,请批评指出!

Activity

  • 定义:Activity是Android的四大组件之一,它是一种可以包含用户界面的组件,主要用于和用户进行交互。
  • 手动创建一个Activity的三个步骤
  • step 1:创建Activity
    在项目目录的 app\src\main\java\包名 右键->New->Activity->Empty Activity


    创建Activity

    先取消Generate Layout File选项的勾选,输入Activity Name ,然后点击 Finish 按钮。这样就完成了第一步。
    补充说明:
  1. 勾选Generate Layout File表示自动为这个Activity创建一个对应的布局文件(默认勾选)。
  2. 勾选Launcher Activity表示自动将这个Activity设置为当前项目的主Activity(每个项目只能有一个主Activity,主Activity即启动app时显示的第一个Activity)。
  3. 勾选Backwards Compatibility(AppCompat)表示会为项目启用向下兼容的模式(默认勾选)。

  • Step 2:创建和加载布局文件
    在项目目录的 app\src\main\res\layout 右键->New->XML->Layout XML File




  • aar android 创建activity android怎么创建activity_android



  • 创建布局文件

    接下来输入Layout File Name,Root Tags 表示布局文件的根布局,这里默认LinearLayout(线性布局),后面会介绍关于布局的内容。
    然后在NewActivity.java文件中为这个活动加载我们创建的布局(添加圈出的代码):




  • aar android 创建activity android怎么创建activity_数据_02



  • 加载布局

    其实就是将我们创建的Activity和布局文件关联起来,这样我们打开这个Activity
    时显示的就是activity_new这个布局文件所编写的内容。
  • Step3:注册Activity
    打开项目目录中 app\src\main\AndroidManifest.xml 文件,在这里为我们创建的Activity进行注册(实际上,在第一步我们创建这个Activity时就已经自动注册了)。


    注册Activity

    看起来这一步似乎是多余的,这是因为Android Studio自动帮我们注册了Activity。但是,如果我们的Activity不是通过上面的步骤创建的,而是从别的地方复制到项目中来的,那么我们就需要谨记要自己在AndroidManifest文件中为这个活动进行注册,否则就会出错。
    这样,我们就完成了手动创建一个Activity的流程。实际上,在第一步我们只要勾选上Generate Layout File 选项,后面两步AndroidStudio都会自动为我们完成。
    补充说明:
    关于主Activity,在AndroidManifest.xml文件(上图)中我们可以看到,在MainActivity的标签下,有这样几行代码:
<intent-filter>
    <action android:name="android.intent.action.MAIN" />
    <category android:name="android.intent.category.LAUNCHER" />
</intent-filter>

实际上,在<activity>标签内部加入<intent-filter>标签,并添加上面的两句声明,就是将这个Activity指定为主Activity。我们可以在创建Activity时勾选上Launcher Activity,Android Studio就会自动为我们配置这个Activity为主Activity,当然,也可以自己在需要指定为主Activity对应的注册标签<activity>中添加上面的代码。

Toast的使用

  • Toast 是Android系统提供的一种非常好的提醒方式,在程序中可以使用它将一些短小的信息通知给用户,这些信息会在一段时间后自动消失,并且不会占用任何屏幕空间。
  • Toast的用法非常简单,通过静态方法makeText()创建出一个Toast对象,然后调用show()方法将Toast显示出来就可以了。
@Override
 public void onClick(View v) {
     switch (v.getId()){
         case R.id.show_toast_button:
             Toast.makeText(SecondActivity.this, "You clicked button",
                     Toast.LENGTH_SHORT).show();
             break;
         default:
             break;
     }
 }

上面的代码表示通过点击我们指定的Button触发Toast弹出,我们可以看到makeText()方法有三个参数。

  1. 第一个参数是Context,即Toast要求的上下文,我们暂且只需要知道Context是一个抽象类,而Activity是它的子类,因此我们指定的SecondActivity.this就是一个Context对象。
  2. 第二个参数是我们要显示的提示信息。
  3. 第三个参数是Toast显示的时长,有两个内置常量Toast.LENGTH_SHORTToast.LENGTH_LONG可以选择。

效果图如下:

aar android 创建activity android怎么创建activity_android_03

效果图.gif

Menu的使用

  • 由于手机屏幕空间十分有限,因此Android提供了Menu,用于展示菜单的同时,也不占用过多屏幕空间。
  • 使用方法:
  • step1:创建菜单布局
    首先在res目录下新建一个menu文件夹,用于存放Menu布局文件,右键res->New->Directory,输入文件夹名menu;然后右键menu->New-Menu resource file,输入文件名,点击OK完成。


    创建Menu布局文件
    接下来根据需要创建菜单项,这里我们创建三个菜单项,代码如下:
<?xml version="1.0" encoding="utf-8"?>
    <menu xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:id="@+id/add_item" android:title="@string/add"/>
    <item android:id="@+id/del_item" android:title="@string/del"/>
    <item android:id="@+id/modify_item" android:title="@string/modify"/>
</menu>
  1. <item>标签就是用来创建具体的某个菜单项;
  2. android:id标签是用来给这个菜单项指定一个唯一的标识符;
  3. android:title标签是给这个菜单项指定一个名称。
  • step2:为当前活动创建菜单——重写onCreateOptionsMenu()方法
    由于我们要给SecondActivity添加Menu,因此我们在SecondActivity.java文件中重写onCreateOptionsMenu()方法。
@Override
public boolean onCreateOptionsMenu(Menu menu) {
   getMenuInflater().inflate(R.menu.second, menu);
   return true;
}
  • 通过getMenuInflater()方法获取到MenuInflater对象,再调用它的inflate()方法,给当前Activity创建菜单。inflate()方法接收两个参数:
  1. 第一个参数 用来指定第一步创建的布局文件;
  2. 第二个参数用于指定我们的菜单项添加到那个Menu对象中,这里自然是onCreateOptionsMenu()方法传入的menu对象,表示当前Activity的Menu。
  3. 返回值为true表示允许菜单显示;否则菜单不显示。
  • step3:为菜单定义响应事件——重写onOptionsItemSelected()方法
    在SecondActivity中重写onOptionsItemSelected()方法,通过item.getItenId()方法来判断我们点击的是哪个菜单项,并为其添加逻辑处理,这里我们弹出一个Toast。
@Override
public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()){
        case R.id.add_item:
            Toast.makeText(SecondActivity.this, "添加", Toast.LENGTH_SHORT).show();
            break;
        case R.id.del_item:
            Toast.makeText(SecondActivity.this, "删除", Toast.LENGTH_SHORT).show();
            break;
        case R.id.modify_item:
            Toast.makeText(SecondActivity.this, "修改", Toast.LENGTH_SHORT).show();
            break;
        default:
            break;
    }
    return true;
}

效果图如下:

aar android 创建activity android怎么创建activity_android_04

效果图

使用Intent实现活动间跳转

  • 显式Intent
    显式Intent的使用十分简单,直接上代码:
@Override
public void onClick(View v) {
    Intent intent = null;
    switch (v.getId()){
        default:
            break;
        case R.id.ll_button:
            intent = new Intent(MainActivity.this,FirstActivity.class);
            startActivity(intent);
            break;
        case R.id.l2_button:
            intent = new Intent(MainActivity.this,SecondActivity.class);
            startActivity(intent);
            break;
    }
}
  • 首先,我们直接new一个Intent对象,构造方法接收两个参数,第一个参数是上下文,因为当前Activity是MainActivity,所以我们传入MainActivity.this,第二个参数指定想要启动的目标活动,这里实际上有两个button,我们分别传入FirstActivity.class和SecondActivity.class,所以点击两个button时会分别跳转到FirstActivity和SecondActivity。下面看效果:

aar android 创建activity android怎么创建activity_开发工具_05

效果图.gif

  • 隐式Intent
    相比于显式Intent,隐式Intent则含蓄了很多,它并明确指出我们要启动那个Activity,而是通过指定一系列action和category等信息,然后由系统进行分析,继而启动合适的活动。
    在这里我用两个Activity进行对比,通过下面的步骤对要启动的Activity进行配置,并实现隐式启动:
  • step1:为Activity配置action和category
    在AndroidManifest文件中,我们给ThirdActivity和NewActivity分别配置<action>标签和<category>标签:
<activity android:name=".ThirdActivity" android:label="@string/thirdactivity">
    <intent-filter>
        <action android:name="com.example.aboutactivity.MY_ACTION"/>
        <category android:name="android.intent.category.DEFAULT"/>
    </intent-filter>
</activity>
<activity android:name=".NewActivity" android:label="@string/new_activity">
    <intent-filter>
        <action android:name="com.example.aboutactivity.MY_ACTION"/>
        <category android:name="android.intent.category.DEFAULT"/>
        <category android:name="android.intent.category.MY_CATEGORY"/>
    </intent-filter>
</activity>
  • step2:给Intent添加<action>和<category>标签
@Override
public void onClick(View v) {
  Intent intent = null;
  switch (v.getId()){
      case R.id.l3_button:
          intent = new Intent("com.example.aboutactivity.MY_ACTION");
          startActivity(intent);
          break;
      case R.id.l4_button:
          intent = new Intent("com.example.aboutactivity.MY_ACTION");
          intent.addCategory("android.intent.category.MY_CATEGORY");
          startActivity(intent);
          break;
      default:
          break;
  }
}
  • 从上面的代码可以看出来,我们给第一个intent添加了“com.example.aboutactivity.MY_ACTION”这个action,给第二个intent添加了“com.example.aboutactivity.MY_ACTION”这个action和“android.intent.category.MY_CATEGORY”这个category,细心的话可能会注意到,两个intent添加的action是相同的,并且第一个intent没有添加category,这是因为“android.intent.category.DEFAULT”这个category是默认category,会自动添加。而且因为我们给第二个intent添加了额外的category标签,因此两个Activity的action和category可以区分开来,这一点是为了说明,我们需要给不同的Activity指定可以区分的<action>标签和<category>标签,否则系统识别不出我们要启动的Activity。下面看效果吧!

aar android 创建activity android怎么创建activity_android_06

隐式Intent.gif

  • 从上面的效果图我们可以看出,由于两个Activity都具有相同的<action>和默认的<category>所以当我们点击ThirdActivity这个button时,系统询问我们要启动哪个Activity,我想这个现象在我们使用手机的时候也遇到过,而由于NewActivity具有自己独有的<category>标签“android.intent.category.MY_CATEGORY”,因此我们点击NewActivity这个bitton时,系统准确的启动了NewActivity,这就是隐式Intent的奇妙之处了!

Activity之间的数据传递

  • 向下一个Activity传递数据
    前面我们知道Intent可以用来启动Activity,其实通过Intent我们还可以向下一个Activity传递数据,Intent提供了一系列putExtra()方法的重载,我们可以把想要传递的数据暂存在Intent中,启动另一个Activity后,我们只需要再将数据取出来,就可以了。接下来我们结合Toast以及前面创建好的MainActivity和SecondActivity来做一个演示。
  • step1:在第一个Activity中将数据暂存到Intent中
    首先我们就在MainActivity中将数据暂存到Intent中,代码如下:
@Override
public void onClick(View v) {
  Intent intent = null;
  switch (v.getId()){
      case R.id.l2_button:
          intent = new Intent(MainActivity.this,SecondActivity.class);
          intent.putExtra("Integer",1024);
          intent.putExtra("String","Hello World!");
          startActivity(intent);
          break;
      default:
          break;
  }
}
  • step2:在启动的Activity中将数据取出
    要取出数据,我们首先要通过getIntent()方法获取到用于启动SecondActivity的Intent对象,然后调用对应数据类型的get方法将数据取出。代码如下:
@Override
public void onClick(View v) {
  switch (v.getId()){
      case R.id.show_toast_button:
          Intent intent = getIntent();
          int num = 0;
          String str = null;
          if(intent != null){
              num = intent.getIntExtra("Integer", 0);
              str = intent.getStringExtra("String");
          }
          Toast.makeText(SecondActivity.this, str+"\t"+num ,
                  Toast.LENGTH_SHORT).show();
          break;
      default:
          break;
  }
}
  • 我们在SecondActivity里面通过一个button触发Toast将传递过来的数据展示出来,下面看效果:

aar android 创建activity android怎么创建activity_移动开发_07

向下一个Activity传递数据.gif

  • 返回数据给上一个活动
    向下一个Activity传递数据十分简单,接下来说一说如何返回数据给上一个Activity。我们可以模拟这样一个场景:在前面我们通过MainActivity启动了SecondActivity,并同时向其传递了一些数据,那么假设我们在SecondActivity中收到数据时,要返回一个接收到数据的确认消息给MainActivity该怎么实现呢?Activity中有一个startActivityForResult()方法,显然这就是我们所需要的。
  • step1:通过startActivityFprResult()方法启动Activity
    我们来修改MainActivity中的代码:
@Override
public void onClick(View v) {
  Intent intent = null;
  switch (v.getId()){
      case R.id.l2_button:
          intent = new Intent(MainActivity.this,SecondActivity.class);
          intent.putExtra("Integer",1024);
          intent.putExtra("String","Hello World!");
          startActivityForResult(intent,1024);
          break;
      default:
          break;
  }
}
  • step2:在启动的Activity中添加返回数据的逻辑
    我们首先在SecondActivity中添加一个button,然后为其添加点击事件。在这里我们要构建一个Intent用于传递数据,将要传递的数据存放到Intent中,然后调用setResult()方法。代码如下:
@Override
public void onClick(View v) {
  switch (v.getId()){
      case R.id.data_back_button:
          Intent intent = new Intent();
          intent.putExtra("Confirm","收到数据");
          setResult(RESULT_OK, intent);
          finish();
          break;
      default:
          break;
  }
}
  • step3:获取返回的数据
    既然数据已经返回到MainActivity了,并且我们现在也回到MainActivity里面了,那么我们怎么知道返回的数据呢?因为我们是通过startActivityForResult()方法启动的SecondActivity,所以当我们再次回到MainActivity时,会回调onActivityResult()方法。也就是说,通过重写onActivityResult()方法,我们可以对返回的数据进行处理。下面看代码:
@Override
protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
  switch (requestCode){
      case 1024:
          if (resultCode == RESULT_OK && data != null ){
              String str = data.getStringExtra("Confirm");
              Toast.makeText(MainActivity.this, str, Toast.LENGTH_SHORT).show();
          }
          break;
      default:
          break;
  }
}
  • onActivityResult()方法的三个参数分别是我们前面传入的请求码、返回数据时传入的处理结果和携带返回数据的Intent对象。可以看到,前面我们提到的请求码和结果码都派了用场,下面看效果:

aar android 创建activity android怎么创建activity_数据_08

向上一个Activity返回数据.gif

Activity实际场景应用

  • 判断当前是在哪个Activity
    当我们在看别人的项目时,经常会遇上这么一个问题:这个app这么多页面,究竟每一个对应哪个Activity呢?其实想要解决这个问题并不难,下面我们来学习一个小技巧。
  • step1:创建一个BaseActivity类
    为什么是BaseActivity类呢?因为这个BaseActivity并不是一个要显示出来的Activity,它只是作为我们Activity的基类,提供一些方法,从而实现我们的目的,并且让它继承自AppCompatActivity。右键包名->New->JavaClass


    创建BaseActivity

    然后重写onCreate()方法,这里为了演示效果,我们用Toast弹出当前Activity实例的类名。(在实际项目中我们用Log.d()打印出来就行了)代码如下:
@Override
 protected void onCreate(@Nullable Bundle savedInstanceState) {
     super.onCreate(savedInstanceState);
     Toast.makeText(BaseActivity.this, 
             getClass().getSimpleName(), Toast.LENGTH_SHORT).show();
 }
  • step2:让项目中所有的Activity都继承自BaseActivity
    由于我们的BaseActivity是继承自AppCompatActivity的,因此这个修改不会带来什么影响。我们直接看运行效果吧!


    知晓当前Activity.gif
  • 随时退出应用
    当我们在写一个由很多Activity组成的项目时,很可能要面临一个问题,从ActivityA打开ActivityB,然后从ActivityB打开ActivityC,这时如果我们按Back键,会依次回退到ActivityB->ActivityA->主界面,按Home键也只是返回桌面,并没有关闭应用。但是如果我们在ActivityC需要直接关闭掉这个应用呢?其实也不难,我们只需要创建一个ActivityCollector类作为Activity管理器,这个问题就迎刃而解啦!
  • step1:创建ActivityCollector类,并添加相关方法实现
public class ActivityCollector {

    private static List<Activity> activities = new ArrayList<>();

    public static void addActivity(Activity activity){
        activities.add(activity);
    }

    public static void removeActivity(Activity activity){
        activities.remove(activity);
    }

    public static void finishAllActivity(){
        for (Activity activity : activities){
            if (!activity.isFinishing()){
                activity.finish();
            }
        }
        activities.clear();
    }
}
  • step2:重写BaseActivity中的相关方法
    既然创建了ActivityCollector类,那么我们每创建一个Activity,就要添加到我们的Activity管理器中,每销毁一个Activity,也要从Activity管理器中将之移除。下面是代码:
public class BaseActivity extends AppCompatActivity {

  @Override
  protected void onCreate(@Nullable Bundle savedInstanceState) {
     super.onCreate(savedInstanceState);
     Toast.makeText(BaseActivity.this,
          getClass().getSimpleName(), Toast.LENGTH_SHORT).show();
     ActivityCollector.addActivity(this);
  }

  @Override
  protected void onDestroy() {
     super.onDestroy();
     ActivityCollector.removeActivity(this);
  }
}
  • step3:在需要直接退出程序的Activity中添加相关逻辑
    例如,我们要在某个Activity中通过点击某个button来关闭应用,那么我们只需要在这个Button的点击事件中调用ActivityCollector.finishAllActivity()就可以啦!关于这个就不写相关示例和演示了,自己动手,印象更深刻!