前言:最近在开发中发现了一个比较严重的问题,当我们将应用按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" >
其中My
Application类就是我们自定义的一个类,继承自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继承类中写一个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类中应当做的是进行一些全局性的配置的初始化操作,而数据的存储应该使用前面推荐的几种方法。