最近根据项目需要,需要做一个切换日志的工具,对于一个app应用已经上线,然后做一个工具能跨应用改变另外一个应用中的值,让日志打印出来,原本想的有两种,一种通过文件的存储与读取来进行判断,另外一种是通过控制该应用的一个方法用sharePreference来设置变量来控制是否打印日志,但这两种都有一个不好的地方就是安全性,别人都可以通过获取存储的文件,同时也可以改变其中的值,所以只能另想它法,最后想到了通过Hook技术以及反射来获取改变另外一个应用中的值,但这种方式需要保证以下几种保持一致:uid一致,pid一致,sharedUserLabel一致,签名sign一致。
manifest配置如下:
- 在
manifest
里面添加:android:sharedUserId
,注意这个属性的取值必须包含点(dot),也就是诸如java packae
的形式。比如com.aaa.bbb
。没有dot的话,将来adb install xxx.apk
就会出错:Failure [INSTALL_PARSE_FAILED_BAD_SHARED_USER_ID]
。最晕死的就是这一点在Android文档中没有提到,感谢万能的google赐予了我答案。所有application都要填写的一样。- 在
manifest
里面添加:android:sharedUserLabel="@string/shared_user_label"
,这个label必须是一个string
资源,不能是raw string
。所有application
都要填写的一样。- 在
manifest
里面添加:android:process="xxx.xxx.xxx"
,这里所有的application
都要填写的一样,内容就是process
的名字,一般来说Android中process
的名字就是manifest
中的package
的取值。- 所有的
application
用同样的一个key来sign
。如果用android studio
开发,由于所有的application
都使用同一个debug key来sign
,所以这一步没有什么额外的工作。更具体的有关sign apk的细节,参考Android文档:http://androidappdocs.appspot.com/guide/publishing/app-signing.html
两个app的application中AndroidManifest.xml都要保持一致,代码如下:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
package="com.example.logswitch"
android:sharedUserId="android.uid.system">
<application
android:name=".MyApplication"
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:sharedUserLabel="@string/shared_user_label"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:process="com.example.demo"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
Android中有Context的概念,想必大家都知道。Context可以做很多事情,打开activity、发送广播、打开本包下文件夹和数据库、获取classLoader、获取资源等等。如果我们得到了一个包的Context对象,那我们基本上可以做这个包自己能做的大部分事情。
Context有个createPackageContext方法,可以创建另外一个包的上下文,这个实例不同于它本身的Context实例,但是功能是一样的。
这个方法有两个参数:
1、
packageName
包名,要得到Context
的包名
2、flags标志位,有CONTEXT_INCLUDE_CODE
和CONTEXT_IGNORE_SECURITY
两个选项。CONTEXT_INCLUDE_CODE
的意思是包括代码,也就是说可以执行这个包里面的代码。CONTEXT_IGNORE_SECURITY
的意思是忽略安全警告,如果不加这个标志的话,有些功能是用不了的,会出现安全警告。
下面给出个例子:
另外一个app应用的包名com.example.demo,其中LogUtil类,有个logSwitch方法,
public class LogUtil {
public static String customTagPrefix = "DemoLog";
public static boolean permitD = true;
public static void logSwitch(Boolean msg) {
if(!msg){
permitD = false;
}else {
permitD = true;
}
}
private static String generateTag(StackTraceElement caller) {
String tag = "%s.%s(Line:%d)";