全面解析Android应用开发中Activity类的用法

Activity类处于android.app包中,继承体系如下:
1.java.lang.Object
2.android.content.Context
3.android.app.ApplicationContext
4.android.app.Activity
activity是单独的,用于处理用户操作。几乎所有的activity都要和用户打交道,所以activity类创建了一个窗口,开发人员可以通过setContentView(View)接口把UI放到activity创建的窗口上,当activity指向全屏窗口时,也可以用其他方式实现:作为漂浮窗口(通过windowIsFloating的主题集合),或者嵌入到其他的activity(使用ActivityGroup)。大部分的Activity子类都需要实现以下两个接口:
onCreate(Bundle)接口是初始化activity的地方. 在这儿通常可以调用setContentView(int)设置在资源文件中定义的UI,使用findViewById(int) 可以获得UI中定义的窗口.
onPause()接口是使用者准备离开activity的地方,在这儿,任何的修改都应该被提交(通常用于ContentProvider保存数据).
为了能够使用Context.startActivity(),所有的activity类都必须在AndroidManifest.xml文件中定义有相关的“activity”项。
Activity类是Android 应用生命周期的重要部分。下面我们具体来总结一下Activity的各个要点:
一、Activity的生命周期

2016229145632612.png (545×711)

二、Activity之间传递数据
1. 通过Intent传递数据
通过Intent.putExtra方法可以将简单数据类型或可序列化对象保存在Intent对象中,然后在另一个Activity中使用getInt、getString等方法获得这些数据。示例代码如下:

Intent intent =new Intent(CurrentActivity.this,OtherActivity.class);
 // 创建一个带“收件人地址”的 email 
 Bundle bundle =new Bundle();// 创建 email 内容
 bundle.putBoolean("boolean_key",true);// 编写内容
 bundle.putString("string_key","string_value"); 
 intent.putExtra("key",bundle);// 封装 email 
 startActivity(intent);// 启动新的 Activity

下面是获取数据:

 Intent intent =getIntent();// 收取 email 
 Bundle bundle =intent.getBundleExtra("key");// 打开 email 
 bundle.getBoolean("boolean_key");// 读取内容
 bundle.getString("string_key");

不过使用Bundle传递数据稍显麻烦,如果你只需要传递一种类型的值可以这样:

Intent intent =new Intent(EX06.this,OtherActivity.class); 
 intent.putExtra("boolean_key",true); 
 intent.putExtra("string_key","string_value"); 
 startActivity(intent);

获取数据:

Intent intent=getIntent(); 
 intent.getBooleanExtra("boolean_key",false); 
 intent.getStringExtra("string_key");

2. 通过静态变量传递数据
适用于不可序列化的且简单的对象,不过不推荐使用静态代码块

public class MainActivity extends Activity { 
  private Button btn; 
  @Override 
  public void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.main); 
    btn = (Button)findViewById(R.id.btOpenOtherActivity); 
    btn.setOnClickListener(new OnClickListener() { 
      @Override 
      public void onClick(View v) { 
        //定义一个意图 
        Intent intent = new Intent(MainActivity.this,OtherActivity.class); 
        //改变OtherActivity的三个静态变量的值 
        OtherActivity.name = "wulianghuan"; 
        OtherActivity.age = "22"; 
        OtherActivity.address = "上海闵行"; 
        startActivity(intent); 
      } 
    }); 
  } 
}
public class OtherActivity extends Activity { 
  public static String name; 
  public static String age; 
  public static String address; 
  private TextView text_name; 
  private TextView text_age; 
  private TextView text_address; 

  @Override 
  public void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.other); 
    text_name = (TextView) findViewById(R.id.name); 
    text_age = (TextView) findViewById(R.id.age); 
    text_address = (TextView) findViewById(R.id.address);   
    //设置文本框的数据 
    text_name.setText("姓名:"+name); 
    text_age.setText("年龄:"+age); 
    text_address.setText("地址:"+address); 
  } 
}

3.通过全局对象传递数据
如果想使某些数据长时间驻留内存,以便程序随时调用,建议采用全局对象。Application全局类不需要定义静态变量只要定义普通成员变量即可,但全局类中必须有一个无参构造方法,编写完Application类后,还需要在<application>标签中制定全局类名,否则系统不会自动创建全局对象。

public class MainApplication extends Application {
  private String username;

  public String getUsername() {
    return username;
  }

  public void setUsername(String username) {
    this.username = username;
  }
}
public class MainActivity extends Activity {

  private MainApplication application;

  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    application = (MainApplication) getApplication();
    application.setUsername("sunzn");
  }

  public void open(View view) {
    Intent intent = new Intent(this,OtherActivity.class);
    startActivity(intent);
  }
}
public class OtherActivity extends Activity {
  private TextView tv_data;
  private MainApplication application;
  private String username;

  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_other);
    application = (MainApplication) getApplication();
    username = application.getUsername();
    tv_data = (TextView) findViewById(R.id.tv_data);
    tv_data.setText("从上一个 Activity 中获取到的数据为:" + username);
  }
}

三、从Activity返回数据
要从Activity返回数据,需要使用startActivityForResult方法来显示Activity。
从Activity1跳转到Activity2:

Intent intent = new Intent();
 intent = intent.setClass(ActivityIntent.this,AnotherActivity.class);
 Bundle bundle = new Bundle();
 bundle.putString("string",et_string.getText().toString());
 intent.putExtras(bundle);
 startActivityForResult(intent,0); //只有这里不同

从Activity2返回数据到Aactivity1:

Intent intent = new Intent();
intent = intent.setClass(AnotherActivity.this,ActivityIntent.class);
Bundle bundle = new Bundle();
bundle.putInt("result","Activity2的处理结果");
intent.putExtras(bundle); 
AnotherActivity.this.setResult(RESULT_OK,intent); //RESULT_OK是返回状态码
AnotherActivity.this.finish();

在Activity1中重写onActivityResault方法,接受数据:

protected void onActivityResult(int requestCode,int resultCode,Intent data) {
          super.onActivityResult(requestCode,resultCode,data);
          switch(resultCode) { //根据状态码,处理返回结果
          case RESULT_OK: 
             Bundle bundle =data.getExtras(); //获取intent里面的bundle对象
               String result = bundle.getInt("result"); 
          break; 
          default:
          break;
          } 

       }

四、Activity的四种创建模式
我们在开发项目的过程中,会涉及到该应用中多个Activity组件之间的跳转,或者夹带其它应用的可复用的Activity。例如我们可能希望跳转到原来某个Activity实例,而不是产生大量重复的 Activity。这样就需要我们为 Activity 配置特定的加载模式,而不是使用默认的加载模式。设置启动模式的位置在 Manifest.xml 文件中 Activity的android:launchMode 属性

1.standard 模式
这是默认模式,无需设置,每次激活Activity时都会创建Activity实例,并放入任务栈中。相当于入栈,按back键返回到前一个Activity相当于退栈。

2.singleTop 模式
如果在任务的栈顶正好存在该Activity的实例,就重用该实例( 会调用实例的 onNewIntent()),否则就会创建新的实例并放入栈顶,即使栈中已经存在该Activity的实例,只要不在栈顶,都会创建新的实例。可用来解决栈顶多个重复相同的Activity的问题

3.singleTask 模式
如果在栈中已经有该Activity的实例,就重用该实例(会调用实例的 onNewIntent() )。重用时,会让该实例回到栈顶,因此在它上面的实例将会被移出栈。如果栈中不存在该实例,将会创建新的实例放入栈中。singleTask 模式可以用来退出整个应用。将主Activity设为SingTask模式,然后在要退出的Activity中转到主Activity,然后重写主Activity的onNewIntent函数,并在函数中加上一句finish。

4.singleInstance 模式
在一个新栈中创建该Activity的实例,并让多个应用共享该栈中的该Activity实例。一旦该模式的Activity实例已经存在于某个栈中,任何应用再激活该Activity时都会重用该栈中的实例( 会调用实例的 onNewIntent() )。其效果相当于多个应用共享一个应用,不管谁激活该 Activity 都会进入同一个应用中。

五、Activity现场保存状态
一般来说,调用onPause()和onStop()方法后的activity实例仍然存在于内存中,activity的所有信息和状态数据不会消失,当activity重新回到前台之后,所有的改变都会得到保留。但是当系统内存不足时,调用onPause()和onStop()方法后的activity可能会被系统摧毁,此时内存中就不会存有该activity的实例对象了. 如果之后这个activity重新回到前台,之前所作的改变就会消失.可以在activity被杀掉之前调用保存每个实例的状态,开发者可以覆写onSaveInstanceState()方法. onSaveInstanceState()方法接受一个Bundle类型的参数,开发者可以将状态数据存储到这个Bundle对象中,以保证该状态在onCreate(Bundle)或者onRestoreInstanceState(Bundle)(传入的Bundle参数是由onSaveInstanceState封装好的)中恢复。这个方法在一个activity被杀死前调用,当该activity在将来某个时刻回来时可以恢复其先前状态。如果调用onSaveInstanceState()方法,调用将发生在onPause()或onStop()方法之前,而且它也不是生命周期的方法。
若向数据库中插入记录等,保存持久化数据的操作应该放在onPause()中. onSaveInstanceState()方法只适合保存瞬态数据,比如UI控件的状态,成员变量的值等.

public class MainActivity extends Activity { 
  private String temp;  
  @Override 
  public void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    // 从savedInstanceState中恢复数据,如果没有数据需要恢复savedInstanceState为null 
    if (savedInstanceState != null) { 
      temp = savedInstanceState.getString("temp"); 
      System.out.println("onCreate: temp = " + temp); 
    } 
  } 

  public void onRestoreInstanceState(Bundle saveInstanceState) { 
    super.onRestoreInstanceState( saveInstanceState); 
    String temp = saveInstanceState.getString("temp"); 
    System.out.println("onResume: temp = " + temp); 

  } 

  // 将数据保存到outState对象中,该对象会在重建activity时传递给onCreate方法和onRestoreInstanceState方法
  @Override 
  protected void onSaveInstanceState(Bundle outState) { 
    super.onSaveInstanceState(outState); 
    outState.putString("temp",temp); 
  }

六、一些关于Activity的技巧
1.设置Activity的方向

android:screenOrientation="portrait">// 竖屏,值为 landscape 时为横屏

2.设置Activity全屏
在onCreate方法中添加如下代码:

// 设置全屏模式
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,WindowManager.LayoutParams.FLAG_FULLSCREEN); 
// 去除标题栏
requestWindowFeature(Window.FEATURE_NO_TITLE);

改变窗口大小、位置、透明度
在onCreate方法中添加如下代码:

Window w=getWindow();
w.setBackgroundDrawableResource(resourceID);//设置窗口背景
WindowManager.LayoutParams layoutParams = w.getAttributes();
layoutParams.height = 200; 
layoutParams.width= 200;
layoutParams.gravity = Gravity.TOP;
layoutParams.x=50;//距离Gravity属性的距离
layoutParams.y=50;
layoutParams.alpha = 0.5;//0:完全透明,1:不透明
w.setAttributes(layoutParams);

3.关闭所有窗口

Intent intent = new Intent(); 
intent.setClass(Android123.this,CWJ.class); 
intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP); //注意本行的FLAG设置 
startActivity(intent)

另一种方法:在调用退出方法中写上MyApplication.getInstance().exit();

public class MyApplication extends Application {

   private List<Activity> activityList = new LinkedList<Activity>();
   private static MyApplication instance;

   private MyApplication() {
   }

   // 单例模式中获取唯一的MyApplication实例
   public static MyApplication getInstance() {
       if (null == instance) {
           instance = new MyApplication();
       }
       return instance;

   }

   // 添加Activity到容器中
   public void addActivity(Activity activity) {
       activityList.add(activity);
   }

   // 遍历所有Activity并finish
   /*
   * 在每一个Activity中的onCreate方法里添加该Activity到MyApplication对象实例容器中
   * 
   * MyApplication.getInstance().addActivity(this);
   * 
   * 在需要结束所有Activity的时候调用exit方法
   * 
   * MyApplication.getInstance().exit();
   */
   public void exit() {

       for (Activity activity : activityList) {
           activity.finish();
       }

       System.exit(0);

   }


版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


更新Android SDK到3.0版本时,遇到Failed to rename directory E:\android\tools to E:\android\temp\ToolPackage.old01问题,导致无法更新,出现该问题的原因是由于3.0版本与较早的sdk版本之间文件结构有冲突,解决
Android 如何解决dialog弹出时无法捕捉Activity的back事件 在一些情况下,我们需要捕捉back键事件,然后在捕捉到的事件里写入我们需要进行的处理,通常可以采用下面三种办法捕捉到back事件: 1)重写onKeyDown或者onKeyUp方法 2)重写onBackPressed方
Android实现自定义带文字和图片的Button 在Android开发中经常会需要用到带文字和图片的button,下面来讲解一下常用的实现办法。一.用系统自带的Button实现 最简单的一种办法就是利用系统自带的Button来实现,这种方式代码量最小。在Button的属性中有一个是drawable
Android中的&quot;Unable to start activity ComponentInfo&quot;的错误 最近在做一款音乐播放器的时候,然后在调试的过程中发现一直报这个错误&quot;Unable to start activity ComponentInfo&quot;,从字面
Android 关于长按back键退出应用程序的实现最近在做一个Android上的应用,碰到一个问题就是如何实现长按back键退出应用程序。在网上查找了很多资料,发现几乎没有这样的实现,大部分在处理时是双击back键来退出应用程序。参考了一下双击back键退出应用程序的代码,网上主流的一种方法是下面
android自带的时间选择器只能精确到分,但是对于某些应用要求选择的时间精确到秒级,此时只有自定义去实现这样的时间选择器了。下面介绍一个可以精确到秒级的时间选择器。 先上效果图: 下面是工程目录: 这个控件我也是用的别人的,好像是一个老外写的,com.wheel中的WheelView是滑动控件的主
Android平台下利用zxing实现二维码开发 现在走在大街小巷都能看到二维码,而且最近由于项目需要,所以研究了下二维码开发的东西,开源的二维码扫描库主要有zxing和zbar,zbar在iPos平台上应用比较成熟,而在Android平台上主流还是用zxing库,因此这里主要讲述如何利用zxing
Android ListView的item背景色设置以及item点击无响应等相关问题 在Android开发中,listview控件是非常常用的控件,在大多数情况下,大家都会改掉listview的item默认的外观,下面讲解以下在使用listview时最常见的几个问题。1.如何改变item的背景色和按
如何向Android模拟器中导入含有中文名称的文件在进行Android开发的时候,如果需要向Android模拟器中导入文件进行测试,通过DDMS下手动导入或者在命令行下通过adb push命令是无法导入含有中文文件名的文件的。后来发现借用其他工具可以向模拟器中导入中文名称的文件,这个工具就是Ultr
Windows 下搭建Android开发环境一.下载并安装JDK版本要求JDK1.6+,下载JDK成功后进行安装,安装好后进行环境变量的配置【我的电脑】-——&gt;【属性】——&gt;【高级】 ——&gt;【环境变量】——&gt;【系统变量】中点击【新建】:变量名:CLASSPATH变量值:……
如何利用PopupWindow实现弹出菜单并解决焦点获取以及与软键盘冲突问题 在android中有时候可能要实现一个底部弹出菜单,此时可以考虑用PopupWindow来实现。下面就来介绍一下如何使用PopupWindow实现一个弹出窗。 主Activity代码:public void onCreat
解决Android中的ERROR: the user data image is used by another emulator. aborting的方法 今天调试代码的时候,突然出现这个错误,折腾了很久没有解决。最后在google上找到了大家给出的两种解决方案,下面给出这两种方法的链接博客:ht
AdvserView.java package com.earen.viewflipper; import android.content.Context; import android.graphics.Bitmap; import android.graphics.BitmapFactory;
ImageView的scaleType的属性有好几种,分别是matrix(默认)、center、centerCrop、centerInside、fitCenter、fitEnd、fitStart、fitXY。 |值|说明| |:--:|:--| |center|保持原图的大小,显示在ImageVie
文章浏览阅读8.8k次,点赞9次,收藏20次。本文操作环境:win10/Android studio 3.21.环境配置 在SDK Tools里选择 CMAKE/LLDB/NDK点击OK 安装这些插件. 2.创建CMakeLists.txt文件 在Project 目录下,右键app,点击新建File文件,命名为CMakeLists.txt点击OK,创建完毕! 3.配置文件 在CMa..._link c++ project with gradle
文章浏览阅读1.2w次,点赞15次,收藏69次。实现目的:由mainActivity界面跳转到otherActivity界面1.写好两个layout文件,activity_main.xml和otherxml.xmlactivity_main.xml&lt;?xml version="1.0" encoding="utf-8"?&gt;&lt;RelativeLayout ="http://schemas..._android studio 界面跳转
文章浏览阅读3.8w次。前言:最近在找Android上的全局代理软件来用,然后发现了这两款神作,都是外国的软件,而且都是开源的软件,因此把源码下载了下来,给有需要研究代理这方面的童鞋看看。不得不说,国外的开源精神十分浓,大家相互使用当前基础的开源软件,然后组合成一个更大更强的大开源软件。好吧,废话不多说,下面简单介绍一下这两款开源项目。一、ProxyDroid:ProxyDroid功能比较强大,用到的技术也比较多,源码也_proxydroid
文章浏览阅读2.5w次,点赞17次,收藏6次。创建项目后,运行项目时Gradle Build 窗口却显示错误:程序包R不存在通常情况下是不会出现这个错误的。我是怎么遇到这个错误的呢?第一次创建项目,company Domain我使用的是:aven.com,但是创建过程在卡在了Building 'Calculator' Gradle Project info这个过程中,于是我选择了“Cancel”第二次创建项目,我还是使用相同的项目名称和项目路_r不存在
文章浏览阅读8.9w次,点赞4次,收藏43次。前言:在Android上使用系统自带的代理,限制灰常大,仅支持系统自带的浏览器。这样像QQ、飞信、微博等这些单独的App都不能使用系统的代理。如何让所有软件都能正常代理呢?ProxyDroid这个软件能帮你解决!使用方法及步骤如下:一、推荐从Google Play下载ProxyDroid,目前最新版本是v2.6.6。二、对ProxyDroid进行配置(基本配置:) (1) Auto S_proxydroid使用教程
文章浏览阅读1.1w次,点赞4次,收藏17次。Android Studio提供了一个很实用的工具Android设备监视器(Android device monitor),该监视器中最常用的一个工具就是DDMS(Dalvik Debug Monitor Service),是 Android 开发环境中的Dalvik虚拟机调试监控服务。可以进行的操作有:为测试设备截屏,查看特定进程中正在运行的线程以及堆栈信息、Logcat、广播状态信息、模拟电话_安卓摄像头调试工具