Resources res = getResources();
Configuration config = res.getConfiguration();
config.locale = locale;
DisplayMetrics dm = res.getDisplayMetrics();
res.updateConfiguration(config, dm);
前两种方法的原理即在应用里实现“选择语言”。通过查看源码,其核心代码为:
IActivityManager iActMag = ActivityManagerNative.getDefault();
try {
Configuration config = iActMag.getConfiguration();
config.locale = locale;
// 此处需要声明权限:android.permission.CHANGE_CONFIGURATION
// 会重新调用 onCreate();
iActMag.updateConfiguration(config);
} catch (RemoteException e) {
e.printStackTrace();
}
PS:感谢 曾阳 的帮助。
可以发现IActivityManager与ActivityManagerNative都是非公开类。如何调用?第一种是API欺骗,第二种是使用Java反射机制。
代码:
ActivityManagerNative.java
package android.app;
/**
* @author Sodino E-mail:[email protected]
* @version Time:2011-7-10 上午11:37:01
*/
public abstract class ActivityManagerNative {
public static IActivityManager getDefault() {
return null;
}
}
IActivityManager.java
package android.app;
import android.content.res.Configuration;
import android.os.RemoteException;
/**
* @author Sodino E-mail:[email protected]
* @version Time:2011-7-10 上午11:37:46
*/
public abstract interface IActivityManager {
public abstract Configuration getConfiguration() throws RemoteException;
public abstract void updateConfiguration(Configuration paramConfiguration)
throws RemoteException;
}
实现模拟了这两个类后,即可正常使用上面提到的转换语系的核心代码了。
2. Java反射机制
不多说了,Java反射机制入门教程:
http://java.sun.com/developer/technicalArticles/ALT/Reflection/index.html
之前写过的几个使用Java反射的例子:
[Android]获取未安装的APK图标(原创非转帖)
http://blog.csdn.net/sodino/article/details/6215224
[Android]挂断、接听电话
http://blog.csdn.net/sodino/article/details/6181610
直接上代码:
private void updateLanguage(Locale locale) {
Log.d("ANDROID_LAB", locale.toString());
try {
Object objIActMag, objActMagNative;
Class clzIActMag = Class.forName("android.app.IActivityManager");
Class clzActMagNative = Class.forName("android.app.ActivityManagerNative");
Method mtdActMagNative$getDefault = clzActMagNative.getDeclaredMethod("getDefault");
// IActivityManager iActMag = ActivityManagerNative.getDefault();
objIActMag = mtdActMagNative$getDefault.invoke(clzActMagNative);
// Configuration config = iActMag.getConfiguration();
Method mtdIActMag$getConfiguration = clzIActMag.getDeclaredMethod("getConfiguration");
Configuration config = (Configuration) mtdIActMag$getConfiguration.invoke(objIActMag);
config.locale = locale;
// iActMag.updateConfiguration(config);
// 此处需要声明权限:android.permission.CHANGE_CONFIGURATION
// 会重新调用 onCreate();
Class[] clzParams = { Configuration.class };
Method mtdIActMag$updateConfiguration = clzIActMag.getDeclaredMethod(
"updateConfiguration", clzParams);
mtdIActMag$updateConfiguration.invoke(objIActMag, config);
} catch (Exception e) {
e.printStackTrace();
}
}
实际运行后,发现对当前系统设置了新的Locale后,不单自己的应用语系改变了,系统所有的应用语系都改变了。这肯定是不合理的。有一个解决办法是在应用界面退出前再次对系统设置成碑的Locale,不过个人不喜欢这样的办法,加之调用updateConfiguration()方法后,整个Activity会重新onCreate(),这个考虑Activity的生命周期可有点费劲了。于是有了下面这第三种方法。
values/strings.xml与xml/english.xml的内容是相同的;values-zh-rCN/strings.xml与xml/chinese.xml的内容也是相同的。出现这样的冗余是因为生成APK时values下的内容都打到rasc去了,读取不了了。
自己实现语系的转换需要考虑到:
3.1 R.xxxxx.id与对应语系中文本串的对应(需要特别考虑到R.array.string字符串数组)。
3.2 解析xml。
3.3 设置语系后,所有界面元素的手动刷新。
在xml中声明一个string是这个的格式:
语言应用
对应R文件会生成一个id指代该string
public static final class string {
public static final int app_name=0x7f050001;
}
Resources res = context.getResources();
String pkg = context.getPackageName();
String tag = "app_name";
int idTag = res.getIdentifier(tag, "string", pkg);
3.2 解析XML