2.3 作为“单例”访问Android应用程序对象
Adrian Cowham
2.3.1 问题
你需要从Android应用程序中访问“全局”数据。
2.3.2 解决方案
最好的解决方案是子类化android.app.Application,把它作为一个有静态存取方法的单例处理。每个Android应用在生命周期中都有一个android.app.Application实例。如果选择子类化android.app.Application,Android将创建类的一个实例,并在其上调用生命周期方法android.app.Application。因为没有什么会阻止你创建子类化的android.app.Application的其他实例,所以它不是真正的单例,但是已经足够接近了。
使会话处理程序、Web服务网关或者应用程序等对象只需要一个实例就可以全局访问,将显著地简化代码。这些对象有时可以当作单例来实现,有时候因为需要Context实例才能正常初始化,因而无法当作单例。在任何一种情况下,在子类化的android.app.Application实例中添加静态存取方法都是有价值的,这样就可以将所有可以全局访问的数据合并在一起,保证对一个Context实例的访问,轻松地编写“正确”的单例代码,而不需要担心同步问题。
2.3.3 讨论
当编写Android应用时,你可能会发现,在多个活动中共享数据和服务是必需的。例如,如果应用具有会话数据(如当前登录用户),你可能希望输出这些信息。在Android平台上开发时,解决这一问题的模式是让android.app.Application实例拥有所有全局数据,然后将Application实例当作单例,用静态存取方法访问各种数据和服务。
当编写Android应用时,你能够保证只有android.app.Application类的一个实例,因此将其作为单例是安全的(Google Android团队也建议这么做)。也就是说,可以在Application实现中安全地添加静态的getInstance()方法。例2-2提供了一个示例。
例2-2:Application实现
public class AndroidApplication extends Application {
private static AndroidApplication sInstance;
private SessionHandler sessionHandler;
public static AndroidApplication getInstance() {
return sInstance;
}
public Session Handler getSessionHandler()
return sessionHandler;
}
@Override
public void onCreate() {
super.onCreate();
sInstance = this;
sInstance.initializeInstance();
}
protected void initializeInstance() {
// do all your initialization here
sessionHandler = new SessionHandler(
this.getSharedPreferences( "PREFS_PRIVATE", Context.MODE_PRIVATE ) );
}
}
这不是经典的单例实现,但是考虑到Android框架的限制,这已经是我们拥有的最接近的解决方案了,它安全而有效。
在这个应用程序中,这种技术的使用简化并整理了实现部分。而且,单例技术使测试的开发更加轻松,结合Robolectric测试框架,你可以很简单地模拟整个执行环境。
还有,不要忘记在AndroidManifest.xml文件中添加应用程序声明:
<application android:icon="@drawable/app_icon"
android:label="@string/app_name"
android:name="com.company.abc.AbcApplication">