首页 > 代码库 > Android应用之——不要将数据存储在Application类中

Android应用之——不要将数据存储在Application类中

前言:最近在开发中发现了一个比较严重的问题,当我们将应用按home键放入后台运行,一段时间后,当我们再次打开应用的时候,十有八九会出现一个NullPointException的空指针异常,根据logcat的日志,就会定位到一个去全局性到变量去,这是什么原因呢?原来,是因为我们我们将很多数据放入了application中作为全局变量,导致了问题的产生,下面来说下为什么不能将数据放在application中。


一、application类的简介
Application和Activity,Service一样是android框架的一个系统组件,当android程序启动时系统会创建一个 application对象,用来存储系统的一些信息。一般情况下,系统会默认帮我们创建一个application类,我们不需要在其中进行任何操作,程序会自动创建。当如果需要创建自己的Application,在其中进行一些操作,也很简单,创建一个类继承 Application并在manifest的application标签中进行注册(只需要给Application标签增加个name属性把自己的 Application的名字定入即可)。
 <application
        android:name="com.example.applicationdemo.MyApplication"
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
其中MyApplication类就是我们自定义的一个类,继承自Application类
public class MyApplication extends Application {	
	public String name;
	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	@Override
	public void onCreate() {
		// TODO Auto-generated method stub
		super.onCreate();
	}	
}

二、application中为什么不能存储数据
下面我们来看一个简单的示例:
我们在application继承类中写一个set get变量的方法,然后通过第一个activity利用application的set方法来设置这个变量的值,在另一个activity中取得这个值,并将其转换为大写显示出来。代码如下:
MainActivity:
public class MainActivity extends Activity {

	private MyApplication application;
	private Button btnName;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.activity_main);
		application = (MyApplication) getApplication();
		application.setName("YangLiang");
		
		btnName = (Button) findViewById(R.id.btn_name);
		
		btnName.setOnClickListener(new OnClickListener() {
			@Override
			public void onClick(View v) {
				Intent intent = new Intent(MainActivity.this, ShowNameActivity.class);
				startActivity(intent);
			}
		});
		
	}
}
MyApplication类
public class MyApplication extends Application {
	public String name;
	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	@Override
	public void onCreate() {
		// TODO Auto-generated method stub
		super.onCreate();
	}
}
另一个activity,显示内容
public class ShowNameActivity extends Activity {

	private MyApplication app;
	private TextView tv;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		// TODO Auto-generated method stub
		super.onCreate(savedInstanceState);
		setContentView(R.layout.showname);
		app = (MyApplication) getApplication();
		String name = app.getName();
		tv = (TextView) findViewById(R.id.tv_showname);
		tv.setText(name.toLowerCase());
	}
	
}
打开应用后,进入ShowNameActivity界面中,然后按home键进入后台,一段时间后,再次打开demo,就出现了空指针异常

logcat中的错误日志

定位到了这句代码:
tv.setText(name.toLowerCase());
name在这个时候已经为空了

这是什么原因呢?
因为当应用程序在后台运行的时候,当我们将应用程序切入到后台或者当内存不足时,系统可能会将当前应用的application进程干掉,
当我们再次从后台将应用切换到前台的时候,系统会重新生成一个application类,这个时候,我们在显示的activity中调用
app = (MyApplication) getApplication();
String name = app.getName();
tv.setText(name.toLowerCase());
上面的name就是空的,因此会出现空指针异常,也就是说,我们存储在application中的数据,有可能会因为程序运行在后台的时候导致application的销毁和重建进而导致数据的丢失,这对于程序来说是致命的,直接导致空指针异常的出现,程序崩溃。


三、有什么更好的办法?
1、通过intent来传递数据,而不是将数据放在全局变量application中,当然这么做是有局限性的,并不是所有地方都适合用intent来传递数据,也并不是所有类型的数据都适合用intent来传递,关于intent的数据传递,读者可参考其他资料。

2、将数据进行持久化操作,写入文件,shareprefrence,数据库等等各种能够安全保存数据的方法。然后在需要使用数据的地方进行文件读取操作。

3、在所有需要使用此类数据的地方进行非空的判断,然后进行相应的操作。

四、总结:不要轻易在application类中进行数据的存储操作,application类中应当做的是进行一些全局性的配置的初始化操作,而数据的存储应该使用前面推荐的几种方法。

demo下载地址:http://download.csdn.net/detail/yanglfree/7709955