简单介绍下这个需求的缘由,这段时间因公司业务需要,其中有一项“设置系统语言”功能,就是在使用app的过程中,动态的去切换整个android机器的语言,具体参照手机设置页面有语言切换功能。起初想来是很简单的事情嘛,不就是个简单的资源国际化嘛,strings.xml资源文件一整还不给ok?真正动起手来就真不是这么一回事了,国际化是没问题,但是怎样能更改所有页面的文字资源呢,这是一个问题。下面介绍下网上找的几个方案。
一、api欺骗
烧制到手机中的android.jar包含了android所需的各种类与方法;而供开发者使用的android.jar只是其中的一部分。api欺骗是指在应用中去模拟未公开的类和方法让应用编译通过并生成apk,然而在应用实际运行中调用的却仍是烧制到手机中真实的android.jar。
二、使用java反射机制
iactivitymanager与activitymanagernative都是非公开类,使用java反射去调用其中的方法。
但是这个弊端是显而易见的,上述两种方法都是去更改系统的语言的类型,功能和你去设置页面去设置语言类型的效果一样。发现对当前系统设置了新的locale后,不单自己的应用语系改变了,系统所有的应用语系都改变了,这正是我们所需要的。
核心代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
|
/** * todo<更新系统语言> * * @author xiho * @versioncode 1 <每次修改提交前+1> */ @suppresswarnings ( "unchecked" ) public class languageutils { public static void updatelanguage(locale locale) { try { object objiactmag, objactmagnative; class clziactmag = class .forname( "android.app.iactivitymanager" ); class clzactmagnative = class .forname( "android.app.activitymanagernative" ); //amn = activitymanagernative.getdefault(); method mtdactmagnative$getdefault = clzactmagnative .getdeclaredmethod( "getdefault" ); objiactmag = mtdactmagnative$getdefault.invoke(clzactmagnative); // objiactmag = amn.getconfiguration(); method mtdiactmag$getconfiguration = clziactmag .getdeclaredmethod( "getconfiguration" ); configuration config = (configuration) mtdiactmag$getconfiguration .invoke(objiactmag); // set the locale to the new value config.locale = locale; //持久化 config.usersetlocale = true; class clzconfig = class .forname( "android.content.res.configuration" ); java.lang.reflect.field usersetlocale = clzconfig .getfield( "usersetlocale" ); usersetlocale.set(config, true ); // 此处需要声明权限:android.permission.change_configuration // 会重新调用 oncreate(); class [] clzparams = { configuration. class }; // objiactmag.updateconfiguration(config); method mtdiactmag$updateconfiguration = clziactmag .getdeclaredmethod( "updateconfiguration" , clzparams); mtdiactmag$updateconfiguration.invoke(objiactmag, config); backupmanager.datachanged( "com.android.providers.settings" ); } catch (exception e) { e.printstacktrace(); } } } |
这样我们利用java的反射机制,调用那些隐藏的方法就可以实现了。
需要注意的是调用此方法:
1
2
|
// objiactmag.updateconfiguration(config); mtdiactmag$updateconfiguration.invoke(objiactmag, config); |
需要加上权限:
1
|
android.permission.change_configuration |
并且此处会重新调用oncreate方法,我就在这个地方处被坑了一把。(如果调用此方法的时候做了一些逻辑,就注意下)。
最后声明:
既然是更改系统的配置当然你的签名也应该是系统签名和shareduserid。不然会类似以下的错误!
error:
java.lang.securityexception: permission denial: updateconfiguration() from pid=31594, uid=10099 requires android.permission.change_configuration
各位都注意下吧~