Android 免Root实现Apk静默安装,覆盖兼容市场主流的98%的机型
地址:http://blog.csdn.net/sk719887916/article/details/46746991
作者: skay
最近在做apk自我静默更新,在获取内置情况下,或者已root过的手机可以完美实现自我静默安装功能,但是发布到市场的apk非内置(非system apk) 也非root,所以自我静默安装做起来不太靠谱。因此借助辅助去实现了一个apk辅助自动装功能,辅助功能可以参考谷歌官网指南:
https://developer.android.com/reference/android/accessibilityservice/package-summary.html
Accessibilityservice
一 简介
accessibilityservice是用户可选服务,
AccessibilityService由系统在后台运行,并接收回调函数AccessibilityEvents。此类事件表示一些状态转换的用户界面,
例如,界面已经改变, 点击一个按钮,等等。这种服务可以选择请求的能力查询活动窗口的内容。
开发一个可访问性服务需要扩展这个类并实现其抽象方法。
AccessibilityService由 AccessibilityServiceInfo来描述。
系统通知的AccessibilityService AccessibilityEvents的节点信息封装在这个类中。
用法
生命周期
AccessibilityService的生命周期管理体系和专门的遵循既定的服务生命周期。
开始触发一个AccessibilityService完全由用户显式地将服务在设备上设置中辅助功能中打开。这样在系统绑定到一个服务,调用callsonServiceConnected()。
此方法可以被重载, 客户想要执行post绑定设置。
AccessibilityService停止或者当用户在设备设置关闭后,会调用disableSelf()。
声明AccessibilityService
AndroidManifest声明AccessibilityService。xml,
但是它必须做两件事:
指定意图处理“android.accessibilityservice.AccessibilityService”。
请求允许BIND_ACCESSIBILITY_SERVICE确保只有系统可以绑定到它。
下面是一个例子声明:
<service android:name=".MyAccessibilityService"
android:permission="android.permission.BIND_ACCESSIBILITY_SERVICE">
<intent-filter>
<action android:name="android.accessibilityservice.AccessibilityService" />
</intent-filter>
. . .
配置
AccessibilityService可以配置为接收特定类型的辅助的事件,监听特定的包,给定的时间内得到每种事件,检索窗口内容, 指定一个设置的activiy,等等。
配置一个可访问性服务有两种方法:
提供元数据条目在清单申报服务。服务声明和一个元数据标记下面:
<service android:name=".MyAccessibilityService">
<intent-filter>
<action android:name="android.accessibilityservice.AccessibilityService" />
</intent-filter>
<meta-data android:name="android.accessibilityservice" android:resource="@xml/accessibilityservice" />
注意:这个方法设置所有属性。
调用setServiceInfo(AccessibilityServiceInfo
)。
注意:这个方法可以调用任何时间动态更改服务配置。
注意:这种方法只允许设置动态可配置属性:eventTypes, feedbackType, flags, notificationTimeout, packageNames
检索视图内容
服务可以指定在其声明, 它可以检索窗口的内容, 返回的为一个AccessibilityWindowInfo对象。
注意:申明此功能要求辅助必须声明其通过SERVICE_META_DATA引用的XML资源配置。
<?xml version="1.0" encoding="utf-8"?>
<accessibility-service
xmlns:android="http://schemas.android.com/apk/res/android"
android:accessibilityEventTypes="typeAllMask"
android:accessibilityFeedbackType="feedbackGeneric"
android:accessibilityFlags="flagDefault"
android:canRetrieveWindowContent="true"
android:description="@string/description_auto_install_accessibility_service"
android:notificationTimeout="100" />
三 用法
onServiceConnected()
服务连接时,也就是第一次打开时调用,这里我们可以初始化常量和标签等
onCreate()
服务创建时调用,初始化一些数据
onDestroy()
服务消亡是,或者用户关闭时,调用,这里我们可以去做些业务相关的释放任务,
onAccessibilityEvent()
监测到内容节点时调用
disableSelf()
自身关闭时主动调用
四 实现思路部分
效果图;
我们可以指定监测的某个包,这里我为了实现自动安装,因此监测com.android.packageinstaller
“包下的com.android.packageinstaller.InstallerActivity
界面。
在遍历到需要的安装按钮调用api来实现自动点击功能,微信抢红包是监测微信红白的activity:
aAccessibilityNodeInfo.performAction(targetAction)
关键部分
重写processAccessibilityEvent方法, 用来获得系统当前的AccessibilityEvent信息,找出要监测的包名,和监测的view类型以及节点内容
是安装apk,并且界面是安装acitvity,文本信息是安装的 我们可以帮他执行点击事件。
代码如下:
private void processAccessibilityEvent(AccessibilityEvent aAccessibilityEvent) {
if(aAccessibilityEvent.getSource() != null) {
String packageName = aAccessibilityEvent.getPackageName().toString();
String className = aAccessibilityEvent.getClassName().toString();
String nodeText = aAccessibilityEvent.getSource().getText() == null ? "" : aAccessibilityEvent.getSource().getText().toString().trim();
if(packageName.equals("com.android.packageinstaller")) {
if(className.equalsIgnoreCase("android.app.AlertDialog")) {
// handleAlertDialog(aAccessibilityEvent, className, nodeText); //should for uninstall
Log.e("test", "onAccessibilityEvent alert dialog");
return;
}
AccessibilityNodeInfo targetNode = extractNode(aAccessibilityEvent, aClassName,
“安装”);
if (targetNode != null) {
int targetAction = AccessibilityNodeInfo.ACTION_CLICK;
if ((aAccessibilityNodeInfo != null) && aAccessibilityNodeInfo.isEnabled()
&& aAccessibilityNodeInfo.isClickable()
&& aAccessibilityNodeInfo.performAction(targetAction)) {
return true;
}
return;
}
return;
}
}
}
五 后续
目前用辅助来实现自动装,用户还是能看到系统安装的界面的,我们则在当前界面添加一个window,用来遮盖系统的安装界面,在安装成功后移除目前的window
,但是会遇到机型问题,有的机子安装界面的activity,有的是dialog, 况且内容也不一样,有安装,有下一步,有我知道了,有同意,因此后期做兼容时废了很大力气,
不仅判断机型还要判断rom版本,此项目中几乎覆盖了市场上主流机型,不兼容的请读者自我加入if语言判断,由于首次点击安装按钮时,用户为打开辅助服务,因此
我做了是否开启本服务判断,如果未打开则跳到设置页面开启此服务,开启了则直接安装,
/**
* isAccessibilitySettingsOn
*/
public static boolean isAccessibilityOn(Context mContext) {
int accessibilityEnabled = 0;
final String service = mContext.getPackageName() + "/" + TamicInstallService.class.getCanonicalName();
try {
accessibilityEnabled = Settings.Secure.getInt(
mContext.getApplicationContext().getContentResolver(),
android.provider.Settings.Secure.ACCESSIBILITY_ENABLED);
Log.v(LOG_TAG, "accessibilityEnabled = " + accessibilityEnabled);
} catch (Settings.SettingNotFoundException e) {
Log.e(LOG_TAG, "Error finding setting, default accessibility to not found: "
+ e.getMessage());
}
TextUtils.SimpleStringSplitter mStringColonSplitter = new TextUtils.SimpleStringSplitter(':');
if (accessibilityEnabled == 1) {
Log.v(LOG_TAG, "***ACCESSIBILITY IS ENABLED*** -");
String settingValue = Settings.Secure.getString(
mContext.getApplicationContext().getContentResolver(),
Settings.Secure.ENABLED_ACCESSIBILITY_SERVICES);
if (settingValue != null) {
mStringColonSplitter.setString(settingValue);
while (mStringColonSplitter.hasNext()) {
String accessibilityService = mStringColonSplitter.next();
Log.v(LOG_TAG, " accessibilityService :: " + accessibilityService + " " + service);
if (accessibilityService.equalsIgnoreCase(service)) {
Log.v(LOG_TAG, "We've found the correct setting - accessibility is switched on!");
return true;
}
}
}
} else {
Log.v(LOG_TAG, "---ACCESSIBILITY IS DISABLED--");
}
return false;
}
Activity代码做了业务判断
installButton.setText("Install apk");
installButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (!TamicInstallService.isAccessibilitySettingsOn(MainActivity.this)) {
// TODO Auto-generated method stub
Intent intent = new Intent(Settings.ACTION_ACCESSIBILITY_SETTINGS);
startActivity(intent);
return;
}
TamicInstallService.setInvokeType(TamicInstallService.TYPE_INSTALL_APP);
TamicWindowManager.makeWatingWiew(MainActivity.this, "安装中...").show();
installApk();
}
});
结束语
通过辅助实现了自动装,root的静默安装请看: Android PackageManager源码浅析以及静默安装实现方式
一文
很多时候我们可以根据用户当前系统的情况分别处理,如果是内置渠道的,直接用静默安装实现,如果非root则申请root权限
在root后可以用静默安装,用户拒绝root情况下,可以采用辅助的自动装来实现apk的快速安装,以用来增大apk升级转换率。
如果的需要增量更新的话,可以用插件免安装实现。
辅助自动装:https://github.com/NeglectedByBoss/Autoinstall
插件免安装:https://github.com/NeglectedByBoss/PluginLoader
静默安装:http://blog.csdn.net/sk719887916/article/details/50314017
Android 免Root实现Apk静默安装,覆盖兼容市场主流的98%的机型的更多相关文章
- android apk静默安装
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/47803149 之前有很多朋友都问过我,在Android系统中怎样才能实现静默安装呢 ...
- Android为TV端助力 apk静默安装
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/47803149 之前有很多朋友都问过我,在Android系统中怎样才能实现静默安装呢 ...
- Android免Root无侵入AOP框架Dexposed
Dexposed框架是阿里巴巴无线事业部近期开源的一款在Android平台下的免Root无侵入运行期AOP框架,该框架基于AOP思想,支持经典的AOP使用场景,可应用于日志记录,性能统计,安全控制,事 ...
- apk 静默安装
老大要我弄个自动更新,要用到静默安装,网上找到了些大拿的代码,我拿去改吧改吧,先贴出来: /** * 软件静默安装 * @param apkAbsolutePath apk文件所在路径 * @retu ...
- APP流氓大法之apk 静默安装
老大要我弄个自动更新,要用到静默安装,网上找到了些大拿的代码,我拿去改吧改吧,先贴出来: /** * 软件静默安装 * @param apkAbsolutePath apk文件所在路径 * @retu ...
- android免root hook框架legend
一.前言 Android中hook框架已经非常多了,最优秀的当属Xposed和Substrate了,这两个框架我在之前的文章都详细介绍过了,不了解的同学,可以转战这里:http://www.wjdia ...
- android免root兼容所有版本ui调试工具
SwissArmyKnife是什么 SwissArmyKnife 是一款方便调试android UI的工具,可以兼容所有android版本,不需要root权限.可以直接在android手机屏幕上显示当 ...
- Dexposed:android免Root无侵入Aop框架
在网上看到了阿里推出的一个android开源项目,名为Dexposed, 是一个Android平台下的无侵入运行期AOP框架.旨在解决像性能监控.在线热补丁等移动开发常见难题,典型使用场景为: AOP ...
- Android照片库选择图片裁剪闪退(兼容小米以及7.0以上机型)
未经允许,禁止
随机推荐
- 剑指Offer——知识点储备-常用算法
剑指Offer--知识点储备-常用算法 快速排序 注:若排序是有序的,采用快排,则退化为冒泡排序. 解决这个问题,采用两个选取基准的方法 (1)随机选取基数(在这个区间内随机取一个数) 出现的恶劣情况 ...
- ROS机器人程序设计(原书第2版)补充资料 (贰) 第二章 ROS系统架构及概念
ROS机器人程序设计(原书第2版)补充资料 (贰) 第二章 ROS系统架构及概念 书中,大部分出现hydro的地方,直接替换为indigo或jade或kinetic,即可在对应版本中使用. 由于工作事 ...
- Linux服务器集群系统(LVS)
from:http://www.linuxvirtualserver.org/zh/lvs1.html#5 本文介绍了Linux服务器集群系统--LVS(Linux Virtual Server)项目 ...
- Android View框架总结(七)View事件分发机制
请尊重分享成果,转载请注明出处: http://blog.csdn.net/hejjunlin/article/details/52282833 View布局告一段落,从本篇开始View事件相关分析, ...
- android 关机充电流程
点击打开链接 0.主要流程 usb插入通过传递cmdline给init解析从而启动充电进程 1. LK lk\app\aboot\aboot.c update_cmdline ---------- i ...
- 【Android应用开发】 推送原理解析 极光推送使用详解 (零基础精通推送)
作者 : octopus_truth 转载请注明出处 : http://blog.csdn.net/shulianghan/article/details/45046283 推送技术产生场景 : -- ...
- C语言--指针函数和函数指针
指针函数和函数指针 指针函数其实是一个简称,是指带指针的函数,它本质上是一个函数,只是返回的是某种类型的指针.其定义的格式为: 类型标识符 *函数名(参数表) 函数指针,从本质上说是一个指针,只是它 ...
- 消息字节——MessageBytes
在tomcat核心处理中有这么一个需求--"为了提高编码性能,对于socket接收到的字节流不马上进行某种编码的转码,而是应该保留字节流的形式,在需要时.在指定编码时才进行转码工作" ...
- 05_NoSQL数据库之Redis数据库:Redis的常用命令,键值相关命令和服务器相关命令
Redis常用命令 Redis提供了丰富的命令对数据库和各种数据库类型进行操作,这些命令可以再Linux终端使用. 键值相关命令: Keys:返回满足给定pattern的所有key 用表达式*表 ...
- 04_NoSQL数据库之Redis数据库:set类型和zset类型
sets类型及操作 Set是集合,它是string类型的无序集合.set是通过hash table实现的,添加,删除和查找复杂度都是0(1).对集合我们可以取并集.交集.差集.通过这些操作我们可 ...