您的位置:首页 > 移动开发

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

2014-08-03 17:20 399 查看

前言:最近在开发中发现了一个比较严重的问题,当我们将应用按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
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: