【声明】
欢迎转载,但请保留文章原始出处→_→
生命壹号:http://www.cnblogs.com/smyhvae/
文章来源:http://www.cnblogs.com/smyhvae/p/3924567.html
【正文】
注:四大组件指的是应用组件:Activity、Service、BroadcastReceiver、ContentProvider;之前的控件指的是UI组件。
博文目录:
- 一、Activity简介
- 二、Activity的状态和生命周期
- 三、Activity的基本用法
- 四、向下一个Activity传递数据
- 五、返回数据给上一个Activity
- 六、Activity运行时屏幕方向与显示方式
- 七、Activity的现场保存
- 八、Activity通过SharedPreferences保存数据
一、Activity简介:
Activity组件是四大组件之一,在应用中一个Activity可以用来表示一个界面, 中文意思也可以理解为“活动” ,即一个活动开始,代表Activity组件启动;活动结束,代表一个Activity的生命周期结束。
一个android应用必须通过Activity来运行和启动,和J2ME 的MIDlet 一样,在android中,Activity的生命周期统一交给系统管理。与MIDlet 不同的是安装在android 中的所有的Activity 都是平等的。
理解以下四个基本概念,将有助于我们更好的了解Activity:
• Application(APP)
• Activity
• Activity栈
• Task
每个Application均占有独立的内存空间。需要注意的是:Application之间虽然相互独立,但APP_1中的Activity与APP_2中的Activity之间可以进行通信(调用、访问等)。
二、Activity的状态和生命周期
1、Activity的状态:
(1)Resumed:Activity对象出于运行状态。一个新Activity 启动入栈后,它在屏幕最前端,处于栈的最顶端,此时它处于可见并可以与用户交互的激活状态。
(2)Paused:另一个Activity位于前端,但是本Activity还可见。
Paused状态常用于:当Activity被另一个透明或者Dialog样式的Activity覆盖时的状态。此时它依然与窗口管理器保持连接,系统继续维护其内部状态,所以它仍然可见,但它已经失去了焦点故不可与用户交互。注:一个Activity出于paused状态时,系统并不会释放资源。释放资源你的操作要靠开发者来完成。
(3)Stopped:另一个Activity位于前端,完全遮挡本Activity。
(4)killed:Activity被系统杀死回收或者没有被启动时。
绘制表格如下:
生命周期函数 |
调用时机 |
举例 |
onCreate |
在Activity对象被第一次创建时调用 |
买车 |
onStart |
当Activity变得可见时调用 |
打火,启动 |
onResume |
当Activity开始准备和用户交互时调用 |
踩油门,驱动汽车前进 |
onPause |
当系统即将启动另外一个Activity之前调用 |
松开油门 |
onStop |
当前Activity变得不可见时调用 |
熄火 |
onDestroy |
当前Activity被销毁之前调用 |
车辆报废 |
onRestart |
当一个Activity再次启动之前调用 |
注:on开头的一般是事件的方法。(引申知识:观察者的设计模式)
2、Activity的生命周期:
详情请见本人的另外一篇博客:Activity的生命周期
生命周期的完整代码如下:
import android.app.Activity;
import android.os.Bundle;
import android.util.Log; public class MainActivity extends Activity { private static final String TAG = "smyhvae"; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Log.d(TAG, "onCreate");
} @Override
protected void onStart() {
super.onStart();
Log.d(TAG, "onStart");
} @Override
protected void onResume() {
super.onResume();
Log.d(TAG, "onResume");
} @Override
protected void onPause() {
super.onPause();
Log.d(TAG, "onPause");
} @Override
protected void onStop() {
super.onStop();
Log.d(TAG, "onStop");
} @Override
protected void onDestroy() {
super.onDestroy();
Log.d(TAG, "onDestroy");
} @Override
protected void onRestart() {
super.onRestart();
Log.d(TAG, "onRestart");
}
}
三、Activity的启动模式:(面试注意)
Activity有四种启动模式:standard、singleTop、singleTask、singleInstance。可以在AndroidManifest.xml中activity标签的属性android:launchMode中设置该activity的加载模式。
- standard模式:默认的模式,以这种模式加载时,每当启动一个新的活动,必定会构造一个新的Activity实例放到返回栈(目标task)的栈顶,不管这个Activity是否已经存在于返回栈中;
- singleTop模式:如果一个以singleTop模式启动的activity的实例已经存在于返回桟的桟顶,那么再启动这个Activity时,不会创建新的实例,而是重用位于栈顶的那个实例,并且会调用该实例的onNewIntent()方法将Intent对象传递到这个实例中;
注:如果以singleTop模式启动的activity的一个实例已经存在于返回桟中,但是不在桟顶,那么它的行为和standard模式相同,也会创建多个实例;
- singleTask模式:这种模式下,每次启动一个activity时,系统首先会在返回栈中检查是否存在该活动的实例,如果存在,则直接使用该实例(会调用实例的onNewIntent()方法),并把这个活动之上的所有活动统统清除;如果没有发现就会创建一个新的活动实例;
- singleInstance模式:总是在新的任务中开启,并且这个新的任务中有且只有这一个实例,并让多个应用共享该栈中的该Activity实例。一旦该模式的Activity的实例存在于某个栈中,任何应用再激活该Activity时都会重用该栈中的实例。其效果相当于多个应用程序共享一个应用,不管谁激活该Activity都会进入同一个应用中。
注:也就是说被该实例启动的其他activity会自动运行于另一个任务中。当再次启动该activity的实例时,会重新调用已存在的任务和实例。并且会调用这个实例的onNewIntent()方法,将Intent实例传递到该实例中。和singleTask相同,同一时刻在系统中只会存在一个这样的Activity实例。(singleInstance即单实例)
注:前面三种模式中,每个应用程序都有自己的返回栈,同一个活动在不同的返回栈中入栈时,必然是创建了新的实例。而使用singleInstance模式可以解决这个问题,在这种模式下会有一个单独的返回栈来管理这个活动,不管是哪一个应用程序来访问这个活动,都公用同一个返回栈,也就解决了共享活动实例的问题。(此时可以实现任务之间的切换,而不是单独某个栈中的实例切换)
1、singleInstance模式详解:
singleInstance模式从字面上看比较难理解,下面通过代码举例来分析。代码如下:
(1)新建三个Activity:FirstActivity、SecondActivity、ThirdActivity。同时,将SecondActivity的启动模式设置为singleInstance。
(2)三个Activity的代码如下:
FirstActivity.java:
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button; public class FirstActivity extends Activity { private Button button1; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d("--->FirstActivity", "返回栈的id是" + getTaskId()); //打印当前返回栈的id
setContentView(R.layout.activity_main);
button1 = (Button) findViewById(R.id.button1);
button1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
startActivity(new Intent(FirstActivity.this, SecondActivity.class));
}
});
} }
上方代码中,在onCreate()方法中打印当前返回栈的id。点击按钮,跳转到SecondActivity。
SecondActivity.java:
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button; public class SecondActivity extends Activity { private Button button2; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d("--->SecondActivity", "返回栈的id是" + getTaskId()); //打印当前返回栈的id
setContentView(R.layout.activity_second);
button2 = (Button) findViewById(R.id.button2);
button2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
startActivity(new Intent(SecondActivity.this, ThirdActivity.class));
}
});
} }
上方代码中,在onCreate()方法中打印当前返回栈的id。点击按钮,跳转到ThirdActivity。
ThirdActivity.java:
import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.widget.Button; public class ThirdActivity extends Activity { private Button button3; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d("--->ThirdActivity", "返回栈的id是" + getTaskId()); //打印当前返回栈的id
setContentView(R.layout.activity_third);
} }
运行程序,在FirstActivity中点击按钮进入SecondActivity中,然后在SecondActivity中点击按钮进入ThirdActivity。后台打印日志如下:
上方日志可以看到:SecondActivity的Task id不同于FirstActivity和ThirdActivity,这说明SecondActivity确实是存放在一个单独的返回栈中的,而且这个返回栈中只有SecondActivity这一个活动。
然后,我们按下Back键进行返回,你会发现ThirdActivity竟然直接返回到了FirstActivity,再按下Back键又会返回到SecondActivity,再按下Back键才会退出程序。解释如下:
FirstActivity和ThirdActivity存放在同一个返回栈里,当在ThirdActivity中按下Back键,ThirdActivity出栈,那么FirstActivity就成为了栈顶活动显示在界面上;然后在FirstActivity界面再次按下Back键,这是当前的返回栈已经空了,于是就显示了另一个返回栈的栈顶活动,即SecondActivity。最后按下Back键,这时,所有的返回栈都已经空了,自然也就退出了程序。
三、Activity的基本用法:
1、隐藏标题栏:
requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.activity_main);
注:第一行代码一定要在第二行代码之前执行。
2、在活动当中使用Toast:
例如点击按钮时,弹出吐司:
button.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(MainActivity.this, "You clicked the Button",Toast.LENGTH_SHORT).show();
}
});
3、启动一个Activity的方法:即在默认启动的Activity中启动另一个Activity
核心代码如下:
Intent intent = new Intent();
intent.setClass(MainActivity.this, SecondActivity.class);
具体过程请参考本人的另一篇博客: 当前Activity跳转到另一个Activity的详细过程
4、隐式Intent的用法:
使用隐式Intent,我们不仅可以启动自己程序内的活动,还可以启动其他程序的活动,这使得Android多个应用程序之间的功能共享成为了可能。比如应用程序中需要展示一个网页,没有必要自己去实现一个浏览器(事实上也不太可能),而是只需要条用系统的浏览器来打开这个网页就行了。
【实例】:打开指定网页。
监听器部分的核心代码如下:
button.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(Intent.ACTION_VIEW);
intent.setData(Uri.parse("http://www.baidu.com"));
startActivity(intent);
}
});
第4行代码:指定了Intent的action是 Intent.ACTION_VIEW,这是一个Android系统内置的动作;
第5行代码:通过Uri.parse()方法,将一个网址字符串解析成一个Uri对象,再调用intent的setData()方法将这个Uri对象传递进去。
详见:《android第一行代码》P48页。
如果要打电话的话,可以使用下面的代码:
Intent intent = new Intent(Intent.ACTION_DIAL);
intent.setData(Uri.parse("tel:10086"));
startActivity(intent);
四、向下一个Activity传递数据:
不同的Activity 实例可能运行在一个进程中,也可能运行在不同的进程中。因此,我们需要一种特别的机制帮助我们在Activity 之间传递消息。Android 中通过Intent 对象来表示一条消息,一个Intent 对象不仅包含有这个消息的目的地,还可以包含消息的内容,这好比一封Email,其中不仅应该包含收件地址,还可以包含具体的内容。对于一个Intent 对象,消息“目的地 ”是必须的,而内容则是可选项。
在上面的实例中通过Activity. startActivity(intent)启动另外一个Activity的时候,我们在Intent类的构造器中指定了“收件人地址”。
Activity传递数据有以下两种方式:
1、【方式一】使用Intent自带的bundle对象
传递数据,代码如下:
这里,我们采用另一种绑定监听时间的方法,即在布局文件中,对Button按钮做如下设置:
<Button
android:id="@+id/button1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_below="@+id/textView1"
android:layout_marginTop="22dp"
android:onClick="gotoSecondActivity"
android:text="启动第二个Activity" />
上方第7行代码就是我们绑定的监听事件,点击按钮,将触发gotoSecondActivity()函数中的代码。紧接着做下面的操作。
在MainActivity中发送数据:
public void gotoSecondActivity(View view){
//创建一个意图
Intent intent = new Intent(MainActivity.this,SecondActivity.class); //第一种方式:使用Intent自带的bundle对象
intent.putExtra("name", "smyhvae");//方法:public Intent putExtra (String name, boolean value)
startActivity(intent);
}
在SecondActivity中接收数据:
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
setContentView(R.layout.second); Intent intent = getIntent();
String name = intent.getStringExtra("name"); TextView textView = (TextView)findViewById(R.id.textView1);
textView.setText("name="+name);
}
或者传递一个对象:
新建一个Student.java的类文件,作为传递的对象:
import java.io.Serializable;
//让这个类序列化
public class Student implements Serializable{
int grade ;
String school;
String address; //将这些变量 变成字符串,方便输出
@Override
public String toString() {
return "Student [grade=" + grade + ", school=" + school + ", address="
+ address + "]";
} }
在MainActivity中发送数据:
public void gotoSecondActivity(View view){
//创建一个意图
Intent intent = new Intent(MainActivity.this,SecondActivity.class); //传递自定义类型(对象)
Student student = new Student();
student.grade = 2;
student.school = "UESTC";
student.address = "chengdu";
intent.putExtra("student", student);//方法:public Intent putExtra (String name, Serializable value) startActivity(intent);
}
在SecondActivity中接收数据:
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
setContentView(R.layout.second); Intent intent = getIntent();
Student student = (Student) intent.getSerializableExtra("student");
TextView textView = (TextView)findViewById(R.id.textView1);
textView.setText(student); System.out.println("SecondActivity-onCreate");
}
【工程文件】
链接:http://pan.baidu.com/s/1jGvEc6q
密码:ic6c
2、【方式二】创建Bundle对象来传递
通过按钮监听事件。核心代码如下:
在MainActivity中发送数据:
//通过这个方法跳转到SecondActivity界面
public void gotoSecondActivity(View view){
//创建一个意图
Intent intent = new Intent(MainActivity.this,SecondActivity.class); //第二种传值方式:创建Bundle对象来传递
Bundle bundle = new Bundle(); //创建bundle的内容
bundle.putString("name", "smyhvae");//编写bundle的内容
bundle.putInt("age", 22);
bundle.putLong("id", 20132224); intent.putExtra("person", bundle);//封装bundle。方法:public Intent putExtra (String name, Bundle value)
startActivity(intent);
}
在SecondActivity中接收数据:
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
setContentView(R.layout.second); //获取上一个Activity传递过来的参数
Intent intent = getIntent();
Bundle bundle = intent.getBundleExtra("person");
String name = bundle.getString("name");
int age = bundle.getInt("age");
//获取上一个Activity传递过来的参数,将接收到的数据输出到TextView当中
TextView textView = (TextView)findViewById(R.id.textView1);
textView.setText("name="+name+";"+"age="+age); System.out.println("SecondActivity-onCreate");
}
【工程文件】
链接:http://pan.baidu.com/s/1ntLqzfN
密码:xzn7
五、返回数据给上一个Activity:
步骤如下:
- 启动带返回结果的MainActivity:
startActivityForResult(Intent intent, int requestCode)
第二个参数为请求码,用于在之后的回调中判断数据的来源
Intent intent = new Intent(MainActivity.this, SecondActivity.class);
startActivityForResult(intent,1);
- 在SecondActivity中通过putExtra放入数据,然后调用以下方法:(非常重要)
setResult(int resultCode, Intent data)
resultCode一般只使用RESULT_OK 或RESULT_CANCELED这两个值,第二个参数则把带有数据的Intent传递回去
Intent intent = new Intent();
intent.putExtra("data_return", "smyhvae");
setResult(RESULT_OK, intent);
finish();
- SecondActicity被销毁之前,会调用上MainActivity的 onActivityResult()方法,所以要重写这个方法:
public void onActivityResult(int requestCode, int resultCode,Intent data)
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
switch (requestCode) {
case 1:
if (resultCode == RESULT_OK) {
String returnedData = data.getStringExtra("data_return");
Log.d("FirstActivity", returnedData);
}
break;
default:
}
}
举例略。
六、Activity运行时屏幕方向与显示方式
1、锁定屏幕方向:横屏/ 竖屏
Android 内置了方向感应器的支持。Android 会根据所处的方向自动在竖屏和横屏间切换。但是有时我们的应用程序仅能在横屏/ 竖屏时运行,比如某些游戏,此时我们需要锁定该Activity 运行时的屏幕方向,<activity>节点的android:screenOrientation属性可以完成该项任务,示例代码如下:
【方法一】在清单文件中配置:
<activity android:name=".EX01"
android:label="@string/app_name"
android:screenOrientation="portrait">
// 值为portrait时强制为竖屏, 值为landscape时强制为横屏
</activity>
【方法二】通过代码实现(一般放在onCreate方法中的前面),如下:
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATTION_LANDSCAPE);
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATTION_PORTRAIT);
这里提一个小知识,Android模拟器中,快捷键"Ctrl+F11/F12"可以实现转屏
2、全屏显示:
可以在其onCreate()方法中添加如下代码实现:
//设置全屏模式
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
//去除标题栏
requestWindowFeature(Window.FEATURE_NO_TITLE);
3、以对话框形式显示Activity:
在清单文件中配置:
<activity
android:name="com.example.smyh004activity02.SecondActivity"
android:label="SecondActivity"
android:theme="@android:style/Theme.DeviceDefault.Dialog">
</activity>
特别关注:Activity的启动模式。见《Android第一行代码》P68页
七、Activity的现场保存:
程序在运行时,一些设备的配置可能会改变,如:横竖屏的切换、键盘的可用性等。这种事情一旦发生,Activity会重新创建。
重新创建的过程如下:
- 在销毁之前,会调用onSaveInstanceState()去保存应用中的一些数据,保存在系统当中;
- 然后调用onDestroy()销毁之前的Activity;
- 最后调用 onCreate()或onRestoreInstanceState()方法去重新创建一个Activity。
现场保存的步骤如下:
(1)在MainActivity中,调用onSaveInstanceState(),即添加如下代码就可以将临时数据保存:
@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
String tempData = "Something you want to save";
outState.putString("data_key", tempData);
Log.i(LOG, "onSaveInstanceState..");
}
(2)数据保存之后,修改MainActivity的onCreate()方法:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Log.i(LOG, "onCreate...");
//步骤2:还原当前activity的状态
if (savedInstanceState != null) {
String tempData = savedInstanceState.getString("data_key");
Log.i(LOG, tempData);
}
}
完整代码如下:
package com.example.smyh004activity03;
import android.os.Bundle;
import android.app.Activity;
import android.content.res.Configuration;
import android.util.Log;
import android.view.Menu;
public class MainActivity extends Activity {
private static final String LOG = "Activity";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Log.i(LOG, "onCreate...");
//步骤2:还原当前activity的状态
if (savedInstanceState != null) {
String tempData = savedInstanceState.getString("data_key");
Log.i(LOG, tempData);
}
}
// 步骤1:活动被销毁之前(如在横竖屏切换时),会触发该方法来保存activity数据
@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
String tempData = "open";
outState.putString("data_key", tempData);
Log.i(LOG, "onSaveInstanceState..");
} }
当手动旋转屏幕后,后台输出结果如下:
上图的日志中,如果把生命周期写完整一点,打印的日志如下:
在旋转屏幕时,如果不想重新创建Activity,我们可以通过清单文件AndroidManifest.xml中android:configChanges来指定的某些属性不发生变化,然后通知程序去调用onConfiguratonChanged()方法主动去改变一些设置(当旋转屏幕的时候)。
清单文件中,指定的常见属性有:
- "keyboard" 键盘发生了改变----例如用户用了外部的键盘
- "keyboardHidden" 键盘的可用性发生了改变
- "orientation" 屏幕方向改变
- "screenSize" 屏幕大小改变
设置代码举例如下:
<activity
android:name="com.example.smyh004activity03.MainActivity"
android:label="@string/app_name"
android:configChanges="orientation|screenSize" >
</activity>
注:符号“|”表示“并且”的意思,这行代码在实际应用中很常见。
接着在上面的java代码的基础之上,添加如下代码:
@Override
public void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
Log.i(LOG, "onConfigurationChanged..");
}
最终,当手动旋转屏幕后,后台输出结果如下:
可以看到,onSaveInstanceState()方法并没有被调用,也就是说,旋转屏幕时,当前Activity并没有被销毁。
八、Activity通过Shared Preferences保存数据:
通常情况下会发生这样的问题,我们在编辑短信的同时有电话打进来,那么接电话肯定是要启动另一个Activiy,那么当前编辑短信的Activity所编辑的信息我们想暂时保存下来,等接完电话后回到该Activity时,可以继续编辑短信。该功能需要如何去实现呢?
其实,SharedPreferences使用xml格式为Android应用提供一种永久的数据存贮方式。对于一个Android应用,它存贮在文件系统的/data/ data/your_app_package_name/shared_prefs/目录下,可以被处在同一个应用中的所有Activity 访问。Android 提提供了相关的API来处理这些数据而不需要程序员直接操作这些文件或者考虑数据同步的问题。
现在就用代码来实现这个功能:
首先使用SharedPreferences这个工具类:
private EditText etMsg ;
private Button sendButton;
private SharedPreferences sp;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); etMsg = (EditText)findViewById(R.id.editText1);
sendButton = (Button)findViewById(R.id.button1); // 获取共享属性操作的工具(文件名,操作模式)
14 sp = This.getSharedPreferences("data", 0);
}
上方第14行代码中,调用的方法是:public SharedPreferences getSharedPreferences (String name, int mode)
其中,第一个参数代表XML文件,如果有这个文件,就会操作这个文件,如果没有这个文件,就会创建这个文件;第二个参数代表一种操作模式,0代表私有。
然后,我们要在onPause()方法里保存数据,之所以在onPause()方法里保存,是因为在所有可能会被内存销毁的生命周期函数中,而onPause()方法最先执行。代码如下:
//在onPause()方法中保存数据
@Override
protected void onPause() {
super.onPause();
String msg = etMsg.getText().toString();
Editor editor = sp.edit();
editor.putString("msg", msg); //执行方法:public abstract SharedPreferences.Editor putString (String key, String value)
editor.commit();
}
将数据保存在msg变量中,然后拿到Editor这个编辑器,给它put进去。当然,这些只是在内存中操作,如果要反映到文件当中,还要执行 commit()方法。
紧接着,我们要在onResume()方法中重新还原数据:(为什么要在这个方法中还原数据,不用我多解释)
@Override
protected void onResume() {
super.onResume();
etMsg.setText(sp.getString("msg", ""));
}
当程序中第一次启动的时候,并没有保存数据,所以返回一个默认的空值。将这个返回的数据放到etMsg控件中就行了。
现在我们运行程序,是可以执行的。
例如,现在编辑内容,然后去别的程序,再回来的时候(就算我们把程序退出了),编辑的内容还依然存在。这个时候,我们打开文件浏览器,发现数据是保存在data-data-android工程的文件夹-shared-prefs目录的data.xml文件当中的,而且是永久保存;所以,当在onResume()方法还原数据之后,我们还要加一部分代码,来删掉这个文件里的内容(无法删除文件本身),不然就会永久保存本地成为垃圾了。代码如下:
protected void onResume() {
super.onResume();
etMsg.setText(sp.getString("msg", ""));
Editor editor = sp.edit();
editor.clear();
editor.commit();
}
总结之后,最终的完整版代码如下:
activity_main.xml文件代码:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".MainActivity" >
<EditText
android:id="@+id/editText1"
android:layout_width="match_parent"
android:layout_height="0dp"
android:gravity="top"
android:layout_weight="1"
android:ems="10" />
<Button
android:id="@+id/button1"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Button" />
</LinearLayout>
MainActivity.java的代码如下:
package com.example.smyh001;
import android.app.Activity;
import android.content.SharedPreferences;
import android.content.SharedPreferences.Editor;
import android.os.Bundle;
import android.view.Menu;
import android.widget.Button;
import android.widget.EditText;
public class MainActivity extends Activity {
private EditText etMsg ;
private Button sendButton;
private SharedPreferences sp; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); etMsg = (EditText)findViewById(R.id.editText1);
sendButton = (Button)findViewById(R.id.button1); // 获取共享属性操作的工具
sp = getSharedPreferences("data", 0);
}
//在onPause()方法中保存数据
@Override
protected void onPause() {
super.onPause();
String msg = etMsg.getText().toString();
Editor editor = sp.edit();
editor.putString("msg", msg);//执行方法:public abstract SharedPreferences.Editor putString (String key, String value)
editor.commit();
} //在onResume()方法中还原数据
@Override
protected void onResume() {
super.onResume();
etMsg.setText(sp.getString("msg", ""));
Editor editor = sp.edit();
editor.clear();
editor.commit();
}
}
运行程序之后,我们在编辑框输入一些文字:
退出程序,然后导出data.xml文件,打开后显示如下:
说明输入的文本被保存在了data.xml文件当中。当我们再回到程序,之前输入的文字会被保留在界面上,而data.xml文件中的文本则会被清空。
代码优化:
上方代码中如果我们在第40行代码的后面加下面这一行代码:
etMsg.setSelection((sp.getString("msg", "")).length());
当返回到原程序时,setSelection方法可将输入光标移动到文本的末尾位置以便继续输入。里面的参数sp.getString("msg", "")是之前所输入的字符串。
到这里为止,Android的基础知识就讲完了,以后会不断完善补充的。
我的公众号
想学习代码之外的软技能?不妨关注我的微信公众号:生命团队(id:vitateam)。
扫一扫,你将发现另一个全新的世界,而这将是一场美丽的意外:
Android组件系列----Activity组件详解的更多相关文章
-
Android application 和 activity 标签详解
extends:http://blog.csdn.net/self_study/article/details/54020909 Application 标签 android:allowTaskRep ...
-
Android之路-------Activity的详解
前言 由于接近放假,公司在赶项目所以前段LP比较忙,没什么时间总结和写博客,只是准备睡觉的时候看看书,每天看的不算多,大概10多页左右吧,不过每天坚持如此的话那也是一个庞大的数字. 今天LP的任务完成 ...
-
Android基础之Activity launchMode详解
本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! Activity的lauchmode,是基础的属性,但也是App优化必须掌握的知识,它约束了Acti ...
-
Android Fragment与Activity通讯详解
与activity通讯 尽管fragment的实现是独立于activity的,可以被用于多个activity,但是每个activity所包含的是同一个fragment的不同的实例. Fragment可 ...
-
Android高效率编码-第三方SDK详解系列(三)——JPush推送牵扯出来的江湖恩怨,XMPP实现推送,自定义客户端推送
Android高效率编码-第三方SDK详解系列(三)--JPush推送牵扯出来的江湖恩怨,XMPP实现推送,自定义客户端推送 很久没有更新第三方SDK这个系列了,所以更新一下这几天工作中使用到的推送, ...
-
Android高效率编码-第三方SDK详解系列(二)——Bmob后端云开发,实现登录注册,更改资料,修改密码,邮箱验证,上传,下载,推送消息,缩略图加载等功能
Android高效率编码-第三方SDK详解系列(二)--Bmob后端云开发,实现登录注册,更改资料,修改密码,邮箱验证,上传,下载,推送消息,缩略图加载等功能 我的本意是第二篇写Mob的shareSD ...
-
Android高效率编码-第三方SDK详解系列(一)——百度地图,绘制,覆盖物,导航,定位,细腻分解!
Android高效率编码-第三方SDK详解系列(一)--百度地图,绘制,覆盖物,导航,定位,细腻分解! 这是一个系列,但是我也不确定具体会更新多少期,最近很忙,主要还是效率的问题,所以一些有效的东西还 ...
-
Flutter学习笔记(15)--MaterialApp应用组件及routes路由详解
如需转载,请注明出处:Flutter学习笔记(15)--MaterialApp应用组件及routes路由详解 最近一段时间生病了,整天往医院跑,也没状态学东西了,现在是好了不少了,也该继续学习啦!!! ...
-
Android 之窗口小部件详解--App Widget
Android 之窗口小部件详解--App Widget 版本号 说明 作者 日期 1.0 添加App Widge介绍和示例 Sky Wang 2013/06/27 1 App ...
随机推荐
-
elasticsearch之节点重启
Elasticsearch节点重启时背后发生的故事有哪些,应该注意哪些配置内容,本篇文章做一个简单的探讨. 节点离开 在elasticsearch集群中,假设NodeA因为种种原因退出集群,在Node ...
- C#--参数数组
-
Java Synchronized的用法
synchronized是Java中的关键字,是一种同步锁.它修饰的对象有以下几种: 1. 修饰一个代码块,被修饰的代码块称为同步语句块,其作用的范围是大括号{}括起来的代码,作用的对象是调用这个代码 ...
-
String框架搭建的基本步骤,及从 IOC &; DI 容器中获取 Bean(spring框架bean的配置)--有实现数据库连接池的链接
Spring框架的插件springsource-tool-suite-3.4.0.RELEASE-e4.3.1-updatesite(是一个压缩包)导入步骤: eclipse->help-> ...
-
OneApm,NewRelic
https://newrelic.com/ http://www.csdn.net/article/2013-03-25/2814631-new-relic-mobile-app-real-time- ...
-
bzoj 3295 [Cqoi2011]动态逆序对(cdq分治,BIT)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3295 [题意] n个元素依次删除m个元素,求删除元素之前序列有多少个逆序对. [思路] ...
-
泛虚拟化技术(以Xen为例)
一.概述 最主要的特点是:修改Guest OS的内核代码.通过修改内核,使Guest OS明白自己是运行在R-1,不要直接去运行特权指令,如果要运行就去Hypercall(主动VMM陷入). ...
-
Mybatis 系列4
上篇系列3中 介绍了properties与environments, 本篇继续讲剩下的配置节点之一:typeAliases. typeAliases节点主要用来设置别名,其实这是挺好用的一个功能, 通 ...
-
java 静态代理总结
首先:定义一个抽象类或借口 例: package JingTai; public interface JingTai1 { public void show(); } 接着:定义一个被代理的类,他需要 ...
-
如何连接oracle 12c可插拔数据库
启动根容器:[oracle@eric ~]$ export ORACLE_SID=cup[oracle@eric ~]$ sqlplus / as sysdbaSQL*Plus: Release 12 ...