1、综述
Activity是Android四大组件(Application Components)之一,简单来说Activity就是平常所见到的用户界面,一般情况下,一个Activity所占的窗口是满屏的,但也可以是一个小的位于其它Activity之上的浮动窗口。一个Android工程至少有一个Activity,Activity上面可布有多个view实例,如文本框、进度条、复选框、按钮等等。多个Activity之间通过Intent来实现跳转。
[1]新建一个activity
(1)要继承(extends)Activity类,复写onCreate(bundle)方法,然后通过" setContentView() "绑定此activity的布局文件XML。
public class secondactivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { // TODO Auto-generated method stub super.onCreate(savedInstanceState); setContentView(R.layout.secondactivity);//绑定布局文件 } }
(2)还必须在AndroidManifest.xml里面注册这个新增加的activity。
<activity android:label="second" android:name=".secondactivity" //activity所在.java类 > </activity>
2、Intent的类来实现屏幕之间的跳转:
[1]显示意图跳转:要求必须知道被激活组件的包和class。
应用场景:在应用程序内部进行跳转。 缺点:耦合性较高。
常规使用方式:
(1)在layout中的" activity_main.xml " 中加入一个id为btn的按钮,另外再创建一个任意的layout(将要跳转到得layout),我取名为second。
(2)在java包中新增" SecondActivity.jar "作为目标Activity,如下:
package com.myandroid; import android.app.Activity; import android.os.Bundle; import android.view.Menu; import android.view.MenuItem; public class SecondActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.second); } }
(3)在AndroidManifest中加入一个新的activity标签,至少要声明一个android:name属性,如下:
<activity android:name=".SecondActivity"> </activity>
(4)在起始Activity中添加跳转,需引入import android.content.Intent;命名空间,如下:
package com.myandroid; import android.app.Activity; import android.os.Bundle; import android.view.View; import android.view.Menu; import android.view.MenuItem; import android.widget.Button; import android.content.Intent; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Button button = (Button) this.findViewById(R.id.btn); button.setOnClickListener(new View.OnClickListener(){ @Override public void onClick(View v) { Intent intent = new Intent(); //实例化Intent类 //第一个参数是Context,第二个参数是跳转Activity的类 intent.setClass(MainActivity.this, SecondActivity.class); //跳转 startActivity(intent); //启动 finish();//停止当前的Activity,如果不写,则按返回键会跳转回原来的Activity } }); } }
另外两种显示方式:
/*** 方法一 ***/ Intent intent1 = new Intent(); //参数一是当前的包名,参数二跳转activity的类名,一定要加上包名 intent1.setClassName("com.myandroid", "com.myandroid.SecondActivity"); startActivity(intent1); /*** 方法二 ***/ Intent intent3 = new Intent(); // 参数一是当前的包名,参数二跳转activity的类名,一定要加上包名 ComponentName component = new ComponentName("com.myandroid", "com.myandroid.SecondActivity"); intent3.setComponent(component); startActivity(intent3);
[2]隐式意图:只需要知道跳转activity的动作和数据,就可以激活对应的组件。
应用场景:在不知道某一应用程序的类名,包名,并且引用不到时。可以跳转到其他应用。带参数访问浏览器的实现。 优点:耦合性较低。
注意:data与type不可以分开写,使用setDataAndType()方法 常规使用方式:
(1)在AndroidManifest中加入一个新的activity标签,要用隐式的意图激活Activity,就要设置动作,类别,数据三个参数,Action及Category的名称是可以自己定义的,只要与openOtherActivity方法中对Intent的设置相一致就可以被过滤器找到并最终激活。如下:
<activity android:name=".SecondActivity"> <intent-filter> <action android:name="com.myandroid.SecondActivity" /> <category android:name="com.myandroid.Default" />
</intent-filter> </activity>
(2)对Intent的设置如下。
/*** 隐式意图***/ Intent intent = new Intent(); //只要动作名称与过滤器中的动作名称一致,就可以匹配,如果匹配成功就可以查看类别及数据 //如果没有类别及数据,那么只会匹配动作 intent.setAction("com.android.SecondActivity"); intent.addCategory("com.android.Default"); //startActivity(intent)为Intent对象注入了一个android.intent.category.DEFAULT类别,必须添加该类别,不然报错找不到activity; //intent.addCategory("android.intent.category.DEFAULT"); startActivity(intent);
运行发现报错:
原因:startActivity(intent)为Intent对象注入了一个android.intent.category.DEFAULT类别,必须添加该类别,不然报错找不到activity;
解决:在AndroidManifest中对应的activity中的intent-filter下,添加对应的<categoryandroid:name="android.intent.category.DEFAULT"/>类别。
此时运行程序,可以跳转成功。
结论:只要Intent设置中的action及category在intent-filter中出现,就可以成功跳转,前提是在没有设置数据参数的情况下。
(3)添加参数的情况:
1)加入的数据是Uri类型的,上面代码的意思是匹配以itcast开头的,主机名为www.itcast.com的数据,后面的路径没有要求:
<activity android:name=".SecondActivity">
<intent-filter>
<action android:name="com.myandroid.SecondActivity" />
<category android:name="com.myandroid.Default" />
<data
android:host="cn.host.android"
android:scheme="host" />
</intent-filter>
</activity>
2)在startActivity(intent)前添加设置data属性,如下:
// 如果在声明activity的时候指定了data属性,就一定要设置他的data属性值,和配置的属性值相等 intent.setData(Uri.parse("host://cn.host.android/my"));
此时运行程序,可以跳转成功。
(4)对数据类型进行设置:
1)在intent-filter中添加设置:<data android:mimeType="image/*" />这句代码的意思是对image及所有的图片数据进行匹配。
2)在Intent对象中进行设置:intent.setType("image/jpeg");
运行发现报错:
原因:添加了setType()方法后,会自动清除intent.setData()方法中的数据设置,所以匹配数据不一致导致了失败,即使将intent.setType("image/jpeg");放在setData()方法之前也是不成功的。
解决方法:把setData()及setType()方法删除,然后,采用Intent对象的setDataAndType(),同时设置数据与类型就可以了,如下。
intent.setDataAndType(Uri.parse("host://cn.host.android/my"),"image/jpeg");
此时运行程序,可以跳转成功。
3、Intent数据传递
应用场景:传递数据,RP计算器。
Bundle:数据捆,打包传输
[1]方式一:
(1)发送页面
Intent intent = new Intent(); intent.setClass(MainActivity.this, SecondActivity.class); Bundle bundle = new Bundle(); //打包发送 bundle.putString("name","123"); //绑定参数 intent.putExtra("maps",bundle); startActivity(intent);
(2)接收页面
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); TextView tv = new TextView(this); Intent intent = this.getIntent(); Bundle bundle = intent.getBundleExtra("maps"); //获取打包数据bundle String name = bundle.getString("name"); //取出需要的数据 tv.setText(name); setContentView(tv);
}
或者
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.second);
TextView txt = (TextView)this.findViewById(R.id.txt);
Intent intent = this.getIntent();
Bundle bundle = intent.getBundleExtra("maps"); //获取打包数据bundle
String name = bundle.getString("name"); //取出需要的数据
txt.setText(name);
}
[2]方式二:创建对象类(javabean)Person,实现Serializable接口
private static final long serialVersionUID = 1L;
(1)发送页面
Intent inent = new Intent();
intent.setClass(this,secondActivity.class);
Person person = new Person(name); //序列化对象传递
intent.putExtra("person',person);
startActivity(intent);
(2)接收页面
TextView tv = new TextView(this);
Intent intent = this.getIntent();
Person p = (Person)intent.getSerializableExtra("person');
String name = p.getName();
4、Intent返回数据
[1]跳转前的Activity:
如果想在Activity中得到新打开Activity关闭后返回的数据,需要使用Activity提供的 startActivityForResult(Intent intent,int requestCode) (startActivity()是context提供的抽象方法,而startActivityForResult方法是Activity自己特有的方法)方法打开新的Activity,新的Activity关闭后会向前面的Activity 传回数据,为了得到传回的数据,必须在" 跳转前页面 "的Activity中重写onActivityResult(int requestCode, intresultCode, Intent data)方法。
实例:
(1)改为用startActivityForResult(Intent intent,int requestCode)跳转;
Button button = (Button) this.findViewById(R.id.btn); button.setOnClickListener(new View.OnClickListener(){ @Override public void onClick(View v) { Intent intent = new Intent(); intent.setClass(MainActivity.this, SecondActivity.class); //第二个参数为请求码,可以根据业务需求自己编号 startActivityForResult (intent, 1); } });
(2)重写onActivityResult(int requestCode, intresultCode, Intent data)方法;
//第一个参数为请求码,即调用startActivityForResult()传递过去的值 //第二个参数为结果码,结果码用于标识返回数据来自哪个新Activity @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if(requestCode == 1 && resultCode == Activity.RESULT_OK)
{
String result = data.getExtras().getString("result");//得到新Activity关闭后返回的数据
TextView tv = (TextView)this.findViewById(R.id.txtBack);
tv.setText(result);
}
}
[2]跳转后的Activity返回数据:
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.second); Button button = (Button) this.findViewById(R.id.back); button.setOnClickListener(new View.OnClickListener(){ @Override public void onClick(View v) { Intent intent = new Intent();//数据是使用Intent返回 intent.putExtra("result", "成功返回结果");//把返回数据存入Intent SecondActivity.this.setResult(RESULT_OK,intent);//设置返回数据 SecondActivity.this.finish();//关闭Activity } }); }
注意:
[1]setResult()方法是Activity的特有方法,第一个参数值RESULT_OK是系统Activity类定义的一个常量,值为-1。
[2]在Activity A中打开一个Activity B,如果要在Activity B关闭的时候给Activity A传一些数据,那么在startActivityForResult (intent, 1)后不可以再使用MainActivity.this.finish();
5、横竖屏切换Activity的生命周期
默认情况下会重新加载activity,容易重新加载数据,造成数据丢失。如游戏。
具体流程:onPause -> onStop ->onDestory -> onCreate ->onStart ->onResume
避免横竖屏切换重新创建activity 在清单文件的activity节点中加入:android:configChanges="orientation|keyboardHidden "
6、任务栈
为了提高用户体验 一个应用程序包含多个activity 栈的顶部的activity会在界面中显示,获取焦点。
任务栈的特点:先进后出,后进先出。
7、AndroidManifest中的activity组件
AndroidManifest文件中含有如下过滤器的activity组件为默认启动组件,应用程序启动时,系统自动调用它。