1. 准备

把某系统双开的两个app的信息进行对比

1.1 目录的对比

1.1.1 data目录对比

原应用:

/data/user/0/com.luoyesiqiu.crackme/files

被复制的应用:

/data/user/999/com.luoyesiqiu.crackme/files

1.1.2 apk所在目录对比

原应用:

/data/app/com.luoyesiqiu.crackme-H1Dvbka0t42rzlCAqSpgHQ==/base.apk

被复制的应用:

/data/app/com.luoyesiqiu.crackme-H1Dvbka0t42rzlCAqSpgHQ==/base.apk

通过对比apk安装目录和数据目录,我们可以知道,该系统的双开是共用同一个apk,但是却拥有独立的数据目录。

1.2 进程信息对比

USER PID PPID VSZ RSS WCHAN ADDR S NAME

u0_a161 30284 918 2276572 48420 SyS_epoll_wait 0 S com.luoyesiqiu.crackme

u999_a161 30311 918 2276572 48004 SyS_epoll_wait 0 S com.luoyesiqiu.crackme

通过查看进程信息,可以知道,这两个应用运行于不同的用户中。

为了实现和它相似的功能,我们做下文的配置。

2. 修改创建用户限制

从Android5.0开始,Android支持创建Profile.默认情况下,系统只允许创建一个新的多开用户,也就是只能双开,但是修改源码可以达到创建多个用户。

修改frameworks/base/services/core/java/com/android/server/pm/UserManagerService.java

的MAX_MANAGED_PROFILES字段,改成自己想要创建的最大用户数,它的默认值是1.

3. 创建用户

创建一个用户即创建一个多开容器,调用createProfile方法,flag传入0x00000020,以创建一个用户并将它开启

private static int getUserIdFromUserInfo(Object userInfo) {

int userId = -1;

try {

Field field_id = userInfo.getClass().getDeclaredField("id");

field_id.setAccessible(true);

userId = (Integer)field_id.get(userInfo);

} catch (Exception e) {

e.printStackTrace();

}

return userId;

}

public boolean startUser(int userId){

Object iActivityManager = null;

try {

iActivityManager = Class.forName("android.app.ActivityManagerNative").getMethod("getDefault").invoke(null);

boolean isOk=(boolean)iActivityManager.getClass().getMethod("startUserInBackground",int.class)

.invoke(iActivityManager,userId);

return isOk;

} catch (Exception e) {

e.printStackTrace();

}

return false;

}

public String createProfile(Context context, String userName, int flag) {

UserManager mUserManager = (UserManager) context.getSystemService(Context.USER_SERVICE);

UserHandle userHandle = UserHandle.getUserHandleForUid(0);

Log.d(TAG,"userHandle = "+userHandle.toString());

try {

int getIdentifier=(int)userHandle.getClass().getMethod("getIdentifier").invoke(userHandle);

Log.d(TAG,"Identifier = "+getIdentifier);

mUserInfo=mUserManager.getClass().getMethod("createProfileForUser",String.class, int.class, int.class)

.invoke(mUserManager

,userName

, flag

,getIdentifier);

if(mUserInfo==null){

Log.d(TAG, "mUserInfo is null!");

return null;

}

int userId = getUserIdFromUserInfo(mUserInfo);

boolean isOk=startUser(userId);

Log.d(TAG, "startUserInBackground() userId = " + userId + " | isOk = " + isOk);

return isOk ? ""+userId : null;

} catch (Exception e) {

e.printStackTrace();

}

return null;

}

注:创建用户的App要在AndroidManifest.xml的manifest节点下加入android:sharedUserId="android.uid.system"字段,加入权限,还要使用系统的platform签名对apk进行签名。

4. 配置系统应用不安装到子用户

默认情况下,在创建一个新用户的时候,系统会给新用户复制一份系统应用,但是在子用户中我们并不需要系统应用,所以我们要在子用户中取消安装这些系统应用。

注:系统应用可以不安装到子用户,但是系统服务一定要安装到子用户,否则,运行在子用户的app可能无法正常运行。

修改frameworks/base/services/core/java/com/android/server/pm/Settings.java

createNewUserLI方法,对系统应用和系统服务是否安装到子用户进行配置。

private final String[] excludeLiStrings={

"android",

"android.ext.services",

"android.ext.shared",

"com.android.bluetooth",

"com.android.htmlviewer",

"com.android.inputdevices",

"com.android.shell",

"com.android.certinstaller",

"com.android.externalstorage",

"com.android.providers.contacts",

"com.android.providers.downloads",

"com.android.providers.media",

"com.android.providers.settings",

"com.android.providers.userdictionary",

"com.android.server.telecom",

"com.android.packageinstaller",

"com.android.settings",

"com.android.providers.telephony",

"com.android.mms.service",

"com.android.webview",

"com.android.location.fused",

"com.android.cts.priv.ctsshim",

"com.android.statementservice",

"com.android.defcontainer",

"com.android.keychain",

"com.android.proxyhandler",

"com.android.dreams.basic",

"com.android.printspooler",

"com.android.pacprocessor",

"com.android.providers.downloads.ui"

};

private boolean isInExcludeList(String pkg){

for(String excludePkg:excludeLiStrings){

if(excludePkg.equals(pkg)){

return true;

}

}

return false;

}

void createNewUserLI(@NonNull PackageManagerService service, @NonNull Installer installer,

int userHandle) {

String[] volumeUuids;

String[] names;

int[] appIds;

String[] seinfos;

int[] targetSdkVersions;

int packagesCount;

synchronized (mPackages) {

Collection packages = mPackages.values();

packagesCount = packages.size();

volumeUuids = new String[packagesCount];

names = new String[packagesCount];

appIds = new int[packagesCount];

seinfos = new String[packagesCount];

targetSdkVersions = new int[packagesCount];

Iterator packagesIterator = packages.iterator();

for (int i = 0; i < packagesCount; i++) {

PackageSetting ps = packagesIterator.next();

if (ps.pkg == null || ps.pkg.applicationInfo == null) {

continue;

}

// Only system apps are initially installed.

//Slog.w(TAG, "User handle:"+userHandle+",pkg name:"+ps.name);

//修改的地方,在列表外的应用不安装到子用户

if(userHandle > 0 && !isInExcludeList(ps.name)){

ps.setInstalled(false, userHandle);

}

else{

ps.setInstalled(ps.isSystem(), userHandle);

}

// Need to create a data directory for all apps under this user. Accumulate all

// required args and call the installer after mPackages lock has been released

volumeUuids[i] = ps.volumeUuid;

names[i] = ps.name;

appIds[i] = ps.appId;

seinfos[i] = ps.pkg.applicationInfo.seinfo;

targetSdkVersions[i] = ps.pkg.applicationInfo.targetSdkVersion;

}

}

for (int i = 0; i < packagesCount; i++) {

if (names[i] == null) {

continue;

}

// TODO: triage flags!

final int flags = StorageManager.FLAG_STORAGE_CE | StorageManager.FLAG_STORAGE_DE;

try {

installer.createAppData(volumeUuids[i], names[i], userHandle, flags, appIds[i],

seinfos[i], targetSdkVersions[i]);

} catch (InstallerException e) {

Slog.w(TAG, "Failed to prepare app data", e);

}

}

synchronized (mPackages) {

applyDefaultPreferredAppsLPw(service, userHandle);

}

}

5. 给非系统用户安装和卸载软件

安装app到指定用户

pm install -t -r --user

-t 允许安装测试应用

-r 替换存在的

--user 指定安装到的用户

注:安装app后要重启默认启动器(Launcher),不然可能会出现奇怪的问题

从指定用户卸载app

pm uninstall --user

6. 设置App默认只安装到主用户

开启子用户后,如果调用adb install或者pm install来安装apk,会把apk安装所有用户。这不是我们想要的,所以,我们修改成执行这些命令时,只把app安装到主用户。

第一步:针对用pm install命令安装apk的方式

frameworks/base/cmds/pm/src/com/android/commands/pm/Pm.java

private static class InstallParams {

SessionParams sessionParams;

String installerPackageName;

//int userId = UserHandle.USER_ALL;

int userId = UserHandle.USER_SYSTEM;

}

第二步:针对用adb install命令安装apk的方式

在旧版本系统上,adb install会调用pm install来安装apk,但在新版的系统上会调用cmd package命令来安装apk。

package命令的具体实现在:

frameworks/base/services/core/java/com/android/server/pm/PackageManagerShellCommand.java

所以,修改以下代码:

private static class InstallParams {

SessionParams sessionParams;

String installerPackageName;

//int userId = UserHandle.USER_ALL;

int userId = UserHandle.USER_SYSTEM;

}

7. 删除用户

adb shell pm remove-user

或者调用以下代码删除

public void deleteUser(Context context,int userId){

UserManager userManager=(UserManager) context.getSystemService(Context.USER_SERVICE);

try {

userManager.getClass().getMethod("removeUser",int.class).invoke(userManager,userId);

} catch (Exception e) {

e.printStackTrace();

}

}

8. 修改用户App右下角标

开启多用户后,如果给多个子用户安装相同的App,它们会显示相同的右下小图标(在源码中被叫做Badge),让我们难以辨别,我们可以让不同的用户显示不同的右下小图标,数字图标就是不错的选择,我们来看看如何修改

frameworks/base/core/java/android/app/ApplicationPackageManager.java的getBadgeResIdForUser方法,返回一个Drawable资源id,作为子用户App的右下小图标

private int getBadgeResIdForUser(int userId) {

// Return the framework-provided badge.

if (isManagedProfile(userId)) {

return com.android.internal.R.drawable.ic_corp_icon_badge;

}

return 0;

}

这个图标是要在右下角的,所以在做图标的时候,要做一张大的图标,它的右下角放着要显示的小图标,其余部分以透明填充。做好图标后要生成svg格式,用Android Studio以Vector Assets导入,大小64x64,导入成功会在项目的res/drawable生成drawable资源文件,把资源文件替换到frameworks/base/core/res/res/drawable/ic_corp_icon_badge.xml,并在frameworks/base/core/res/res/values/symbols.xml添加对drawable文件的声明

9. 在最新任务列表出现多开应用

默认情况下,最近任务列表是不会出现多开应用的。

在frameworks/base/services/core/java/com/android/server/am/ActivityManagerService.java的getRecentTasks方法中,有一段校验:

for (int i = 0; i < recentsCount && maxNum > 0; i++) {

TaskRecord tr = mRecentTasks.get(i);

//....

if (!tr.mUserSetupComplete) {

// Don't include task launched while user is not done setting-up.

if (DEBUG_RECENTS) Slog.d(TAG_RECENTS,

"Skipping, user setup not complete: " + tr);

continue;

}

//....

res.add(rti);

//....

}

可以将这段校验注释掉,tr是frameworks/base/services/core/java/com/android/server/am/TaskRecord.java类实例,mUserSetupComplete赋值如下:

mUserSetupComplete = Settings.Secure.getIntForUser(mService.mContext.getContentResolver(),

USER_SETUP_COMPLETE, 0, userId) != 0;

10. 修改多开应用最近任务的名称

子用户默认会在最近任务的应用名称前加上"工作"这两个字,语言是英文会显示"Work",如果想隐藏它们

中文:

frameworks/base/core/res/res/values-zh-rCN/strings.xml

英文:

frameworks/base/core/res/res/values/strings.xml

修改managed_profile_label_badge字段,去掉"工作"或者"Work"即可。

11. 修改卸载时的提示文本

如果是多开应用,卸载时提示的文本和主用户是不一样的,如果需要修改,则修改下面的文件

中文:

packages/apps/PackageInstaller/res/values-zh-rCN/strings.xml

英文:

packages/apps/PackageInstaller/res/values/strings.xml

修改uninstall_application_text_user字段的值

12. 参考

java双开_修改Android源码实现原生应用双开,应用多开相关推荐

  1. 安卓java修改热点_修改Android源码实现连接指定BSSID的热点

    1 概述 需要实现一个需求,Android手机A发射热点,获取BSSID,发送到另一台Android设备,在环境中存在多个同样SSID和密码的情况下,设备只会连接手机A. 1.1 概念 服务集识别码( ...

  2. 关于在Ubuntu中修改Android源码的一些总结

    关于在ubuntu中修改Android源码的一些总结 1.关于配置VMware,网上有一大堆的方法,这里就不详述了,我现在用的源码是4.0.3的.要想改源码,你需要以下基本的东西: 推荐配置: OS ...

  3. java社交婚恋交友app源码 安卓原生app+后台Spri ngMVC+Mybatis+mysql

    java社交婚恋交友app源码 安卓原生app+后台Spri ngMVC+Mybatis+mysql

  4. Android源码编译原生模拟器

    Android源码编译原生模拟器 源代码下载 安装repo 确保主目录下有一个 bin/ 目录,并且该目录包含在路径中: mkdir ~/bin PATH=~/bin:$PATH 下载 Repo 工具 ...

  5. java 事件分发机制_读Android源码之事件分发机制最全总结

    原标题:读Android源码之事件分发机制最全总结 本文源码来自andorid sdk 22,不同版本会有细微差别,但核心机制是一致的 一.概述 事件分发有多种类型, 本文主要介绍Touch相关的事件 ...

  6. 关于在ubuntu中修改Android源码的一些细节

    1.关于配置VMware,网上有一大堆的方法,这里就不详述了,我现在用的源码是4.0.3的.要想改源码,你需要以下基本的东西: 推荐配置: OS Ubuntu 11.04 64bit Gcc 4.5. ...

  7. 安卓java编辑器eclipse_Eclipse开发Android源码的详细教程

    用eclipse + ADT作为android开发工具,可以说是很方便的,在HelloActivity小程序里我们就感觉到eclipse功能的强大. 1.可以使用eclipse来编辑JAVA程序.检查 ...

  8. java 直播_一对一直播源码开发过程中区分Java和PHP的重要性

    开发一对一直播源码光有热情可是不够的,还需要对技术有点了解.除了懂点基本的后台操作外,得先明白一对一直播源码选择Java还是PHP?两者之中那个会更好.今天同创小编主要就是给大家介绍一下两者之间的区别 ...

  9. linux3.10.46源码下载,Ubuntu11.10下编译android源码4.0.3

    1.初始化编译环境 软件需求: Python2.5-2.7,可以http://www.python.org/download/下载(Ubuntu11.10自带). JDK6:编译Gingerbread ...

  10. mac下调试android源码,MacBooK pro调试Android 8.1源码

    工具的选择 1.Android Studio(能够看,能够调试) 2.understand (只适合看)html macbook pro基本上只有这两款能够选择,刚开始用了understand发现并不 ...

最新文章

  1. sql常用crud命令
  2. 有向图生成树是如何画的_漫画:什么是最小生成树?
  3. senborn绘制混淆矩阵
  4. Angular4.x+Ionic3 踩坑之路之打包时出现JAVASCRIPT HEAP OUT OF MEMORY的几种解决办法
  5. Python判断素数(质数)
  6. Kaggle 数据清洗挑战 Day 2 - 数据缩放及标准化处理
  7. [Swift]LeetCode862. 和至少为 K 的最短子数组 | Shortest Subarray with Sum at Least K
  8. Tomcat的作用(自用)
  9. 在登录页面中输入正确的信息还是显示用户名或密码错误
  10. 计算机剪切全选快捷键,全选快捷键是什么
  11. checkbox全选、清除、反选
  12. 【HNOI模拟By lyp】Day1
  13. 《评人工智能如何走向新阶段》后记(再续22)
  14. python编程1-win7上运行python
  15. 美国谍梦第三至五季/全集The Americans迅雷下载
  16. C语言--第三次作业
  17. 一个阿里P8的程序员,一年能赚多少钱?
  18. python将图片批量裁剪成圆形
  19. boost asio 学习
  20. input/print,pycharm设置文件抬头

热门文章

  1. P2P追债也用上大数据
  2. fastreport 横向分栏_fastreport分栏分组显示问题(急贴盼解决)
  3. linux mint 使用软件管理器安装软件
  4. Uva_156 Ananagrams ( map 的应用 )
  5. 何为“定向融资计划”?适合你投资吗?
  6. 电脑出问题解决办法(WinXP)
  7. PyQt4转PyQt5心得
  8. excel表格斜线_WORD圆角表格,如此惊艳
  9. PB调用C#动态库轻松实现微信、支付宝支付
  10. 面向对象系列(四)-接口的特点