昨天组里的同事遇到一些切换多语言的细节问题,发现在Android N版本上配置应用内多语言没有生效,正好拿出来做为一个面试题讲解一下。
面试题:如何实现应用内多语言切换?
我们知道Android的多语言实现很简单,可以在不同的语言环境下使用不同的资源。在不同的res/value-xx下放置不同语言的strings.xml实现字符的本地化,而这个value-xx目录的选择是根据Resource中的Configuration.Locale这项的值来决定的。如zh中文,就会选择value-zh目录,如果没有匹配到(即APK中没有value-zh目录)就使用默认的value目录中的字符资源。
其实最终实现字符串的选择都是在Assets这个类中,通过Native的方法来加载相应的字符串资源。
然而,我们还是会有一些业务场景需要不根据Android系统的Locale配置就改变应用的语言。实现的方式也很简单,直接调用Android开放的接口Resources.updateConfiguration:
public static void changeSystemLanguage(Context context, String language) {
if (context == null || TextUtils.isEmpty(language)) {
return;
}
Resources resources = context.getResources();
Configuration config = resources.getConfiguration();
if (Locale.SIMPLIFIED_CHINESE.getLanguage().equals(language)) {
config.locale = Locale.SIMPLIFIED_CHINESE;
} else {
config.locale = new Locale(language);
}
resources.updateConfiguration(config, null);
}
上面的代码,我们可以在应用内通过language的值指定是显示哪种语言,当然language值我们需要保存在Preferences或者数据库中。
好像很简单,我们的项目为什么还会出现问题呢?而且大家都不知道如何下手,因为在Android N之前的版本都是可以正常切换语言的。后来我跟了一下,发现在MainActivity和SplashActivity这些Activity有继承了自已扩展的BaseActivity,而这个BaseActivity有这样一段代码:
@Override
public Resources getResources() {
Resources res = super.getResources();
Configuration config = new Configuration();
config.setToDefaults();
res.updateConfiguration(config, res.getDisplayMetrics());
return res;
}
config.setToDefaults会将Locale的值设为null,而再调用updateConfiguration可能会根据Android系统的语言重新设置Resources中的Locale。好吧,只是假设,还没有看到Android N的源代码。不过去掉这段代码后,在Android N(Preview)手机上切换语言正常了。
小结
今天遇到的问题,是以前遗留的代码埋下的坑终于暴露出来,也是这个项目缺乏代码审查(Code Review)机制的结果。找了几个人也无法说清覆写getResources这个方法的用意,最终也只能按历史问题处理了,是历史总有一些说不清楚的事,对吧。
回到这个面试题,现在你知道了可以在应用内切换语言(当然也可以修改Configuration的其他值),那么你有没有想过,如果不知道这个updateConfiguration的存在,你会怎么实现这个需求呢?或者说没有人和你说过updateConfiguration,你能找到它吗?
网友评论
以前的做法:
config.setToDefaults会将Locale的值设为null,而再调用updateConfiguration会根据Android系统的之前的语言重新设置Resources中的Locale;
if (o.locale != null) {
locale = (Locale) o.locale.clone();
}
sdk-25的做法:
config.setToDefaults会将Locale的值设为null,而再调用updateConfiguration会设置为null
locale = o.locale == null ? null : (Locale) o.locale.clone();
设置语言,25之前的可以config.setToDefaults();之后res.updateConfiguration(config, res.getDisplayMetrics());可以正常切换语言,但是25之后的不可以,
因为config.setToDefaults();之后,config的local为null,这是一样的,变了的是,res.mTmpConfig.setTo(config); config.setTo()方法,local==null的时候是赋值为null,以前是==null的时候是不处理的,赋值为null之后,之后源码还会判断说,如果locales是空的话,就赋值一个默认的给它,因为手机默认是中文,想给APP设英文就会失败