AndFix简单集成实现
1,下载
https://github.com/alibaba/AndFix
2,将源码拷贝到项目中,或者使用添加依赖包,都可以
添加到项目中的效果如下
3,在application中初始化AndFix
public class BaseApplication extends Application { private PatchManager patchManager; private static BaseApplication instant; public static BaseApplication getInstant() { return instant; } public PatchManager getPatchManager() { return patchManager; } @Override public void onCreate() { super.onCreate(); instant = this; // 初始化热更新数据 try { patchManager = new PatchManager(this); patchManager.init(PackageUtils.getAppVision(this,getPackageName())); patchManager.loadPatch(); }catch (Exception e) { e.printStackTrace(); } } }
public class PackageUtils { public static String getAppVision(Context context,String packageName){ try { String versionName = context.getPackageManager().getPackageInfo(packageName, 0).versionName; return versionName; } catch (PackageManager.NameNotFoundException e) { e.printStackTrace(); } return ""; } }
4,加载差异包也就是 .apatch文件
public class MainActivity extends MPermissionsActivity { private TextView testData,testData2; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); // 6.0 以上读写权限控制 initPermissions(); testData = findViewById(R.id.textView); testData2 = findViewById(R.id.textView2); // 跳转到效果界面查看修改效果 testData.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent = new Intent(MainActivity.this, SecondeActivty.class); startActivity(intent); } }); // 加载差异包文件 testData2.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { // 创建目录 File files = new File(Environment.getExternalStorageDirectory(),"power"); if (!files.exists()) { files.mkdirs(); } // 加载差异包{adb push (.apatch路径) /sdcard/power/a1.apatch } File file = new File(Environment.getExternalStorageDirectory().getAbsolutePath() + "/power/" + "a1.apatch"); if (file.exists()) { try { Log.e("qqqqqqqq","加载差异包"); BaseApplication.getInstant() .getPatchManager() .addPatch(Environment.getExternalStorageDirectory().getAbsolutePath() + "/power/" + "a1.apatch"); } catch (IOException e) { Log.e("qqqqqqqq","加载差异包异常"); e.printStackTrace(); } }else { Log.e("qqqqqqqq","文件不存在!"); } } }); } private void initPermissions() { requestPermission(new String[]{Manifest.permission.READ_EXTERNAL_STORAGE , Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1000); } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); switch (requestCode) { case 1000: boolean cameraAccepted = grantResults[0] == PackageManager.PERMISSION_GRANTED; if (cameraAccepted) { // 授予权限成功 Toast.makeText(MainActivity.this, "授权成功!",Toast.LENGTH_SHORT); } else { Toast.makeText(MainActivity.this, "授权失败!",Toast.LENGTH_SHORT); } break; } } }
网上找的权限封装类
public abstract class MPermissionsActivity extends FragmentActivity { private final String TAG = "MPermissions"; private int REQUEST_CODE_PERMISSION = 0x00099; /** * 请求权限 * @param permissions 请求的权限 * @param requestCode 请求权限的请求码 */ public void requestPermission(String[] permissions, int requestCode) { this.REQUEST_CODE_PERMISSION = requestCode; if (checkPermissions(permissions)) { permissionSuccess(REQUEST_CODE_PERMISSION); } else { List<String> needPermissions = getDeniedPermissions(permissions); ActivityCompat.requestPermissions(this, needPermissions.toArray(new String[needPermissions.size()]), REQUEST_CODE_PERMISSION); } } /** * 检测所有的权限是否都已授权 * * @param permissions * @return */ private boolean checkPermissions(String[] permissions) { if (Build.VERSION.SDK_INT < Build.VERSION_CODES.M) { return true; } for (String permission : permissions) { if (ContextCompat.checkSelfPermission(this, permission) != PackageManager.PERMISSION_GRANTED) { return false; } } return true; } /** * 获取权限集中需要申请权限的列表 * * @param permissions * @return */ private List<String> getDeniedPermissions(String[] permissions) { List<String> needRequestPermissionList = new ArrayList<>(); for (String permission : permissions) { if (ContextCompat.checkSelfPermission(this, permission) != PackageManager.PERMISSION_GRANTED || ActivityCompat.shouldShowRequestPermissionRationale(this, permission)) { needRequestPermissionList.add(permission); } } return needRequestPermissionList; } /** * 系统请求权限回调 * * @param requestCode * @param permissions * @param grantResults */ @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); if (requestCode == REQUEST_CODE_PERMISSION) { if (verifyPermissions(grantResults)) { permissionSuccess(REQUEST_CODE_PERMISSION); } else { permissionFail(REQUEST_CODE_PERMISSION); showTipsDialog(); } } } /** * 确认所有的权限是否都已授权 * * @param grantResults * @return */ private boolean verifyPermissions(int[] grantResults) { for (int grantResult : grantResults) { if (grantResult != PackageManager.PERMISSION_GRANTED) { return false; } } return true; } /** * 显示提示对话框 */ private void showTipsDialog() { new AlertDialog.Builder(this) .setTitle("提示信息") .setMessage("当前应用缺少必要权限,该功能暂时无法使用。如若需要,请单击【确定】按钮前往设置中心进行权限授权。") .setNegativeButton("取消", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { } }) .setPositiveButton("确定", new DialogInterface.OnClickListener() { @Override public void onClick(DialogInterface dialog, int which) { startAppSettings(); } }).show(); } /** * 启动当前应用设置页面 */ private void startAppSettings() { Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS); intent.setData(Uri.parse("package:" + getPackageName())); startActivity(intent); } /** * 获取权限成功 * * @param requestCode */ public void permissionSuccess(int requestCode) { Log.d(TAG, "获取权限成功=" + requestCode); } /** * 权限获取失败 * @param requestCode */ public void permissionFail(int requestCode) { Log.d(TAG, "获取权限失败=" + requestCode); } }
效果展示界面
public class SecondeActivty extends Activity { private TextView textView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_seconde); textView = findViewById(R.id.seconde_textView); init(); } private void init() { // String Tag = "待修复bug的数据?"; String Tag = "已修复bug的数据!"; textView.setText(Tag); } }
记得要添加混淆
#官方AndFix混淆 -keep class * extends java.lang.annotation.Annotation -keepclasseswithmembernames class * { native <methods>; }#AndFix包不要混淆 -dontwarn com.alipay.euler.andfix.** -keep class com.alipay.euler.andfix.**{*;}
准备工作完毕;下面就是要生成差异包了
然后push到sd卡上面,这里是为了展示使用所以放到sd卡上面
最后安装有bug的包就可以实现热修复
demo下载
https://download.csdn.net/download/qq_34601429/10380329
AndFix简单集成实现相关推荐
- 超简单集成HMS ML Kit二代身份证识别,一键实名认证
前言 就在近期华为HMS ML Kit 发布了1.0.3.30版本,ML Kit在原有通用OCR功能的基础上,又新增了银行卡识别(BCR)和二代身份证识别(ICR).今天小编就给大家介绍一下其中的IC ...
- 超简单集成HMS ML套件二代身份证识别,一键实名认证
前言 就在近期华为HMS ML Kit发布了1.0.3.30版本,ML Kit在原有通用OCR功能的基础上,又添加了银行卡识别(BCR)和二代身份识别(ICR).今天小编就给ML kit除了提供语言相 ...
- 环信3即时通信——web端sdk简单集成(一)(文本消息)
环信3即时通信--web端sdk简单集成 实现简单的登录/文字发送/获取好友列表/接收文字消息,(本文章只为简单未使用vuex,消息记录保存在本地localstorage里面) 一.安装sdk npm ...
- 超简单集成ML kit 实现听写单词播报
背景 相信我们大家在刚开始学习一门语言的时候都有过听写,现在的小学生学语文的时候一项重要的课后作业就是听写课文中的生词,很多家长们都有这方面的经历.不过一方面这种读单词的动作相对简单,另一方面家长 ...
- iOS开发之百度地图的简单集成——标注POI检索
iOS开发之百度地图的简单集成--标注&POI检索 .h文件 // Created by XK_Recollection on 16/6/15. // Copyright © 2016年 GN ...
- LTS简介以及与SpringBoot的简单集成
LTS简介以及与SpringBoot的简单集成 一 什么是LTS 关于定时任务,虽然Spring提供了基于注解@EnableScheduling @Scheduled的实现方式.其实现是通过线程池Sc ...
- 基于EasyNVR摄像机网页无插件直播服务二次开发实现H5播放页面的简单集成方案...
我们通常在构架一套视频SaaS应用的过程中,将平台设计为3层:视频硬件层(视频源).视频能力平台(vPaaS).视频应用平台(vSaaS),视频硬件包括各种IPC.NVR.编码器等视频生成设备,vPa ...
- Shiro和SpringBoot简单集成
Shiro是一种简单的安全框架,可以用来处理系统的登录和权限问题. 本篇记录一下Spring Boot和Shiro集成,并使用Jwt Token进行无状态登录的简单例子. 参考Demo地址,此Demo ...
- 【转载】超简单集成HMS ML Kit 人脸检测实现可爱2D贴纸
文章目录 前言 场景 开发前准备 在项目级gradle里添加华为maven仓 在应用级的build.gradle里面加上SDK依赖 在AndroidManifest.xml文件里面申请相机.访问网络和 ...
最新文章
- [bzoj2301][HAOI2011]Problem b
- idea每次新建项目都要重新配置maven
- ai人工智能收入_人工智能促进收入增长:使用ML推动更有价值的定价
- 算法设计与分析——回溯法——装载问题
- ajax的url怎么将后缀补上_蜂蜜杏仁怎么做?杏仁和蜂蜜腌制方法
- 原 剑指offer(刷题11-20)--c++,Python版本
- 电动车式的爱情,有你有我有爱、青春不再,人生无悔!
- SSL 延迟与 Http、Https
- (转)详解Windows Hash
- Day001 20210206
- Pandas深入浅出
- AutoCAD DWG,DXF文件导出高清图片、PDF
- 【安卓项目】期末大作业——“记账宝”APP开发案例
- Unable to negotiate with port 51732: no matching host key type found. Their offer:
- linux互信文件权限,Linux如何建立互信
- html艺术字在线制作,超全的免费图标字体(在线图标字体制作)
- vios配置的自动采集_VIOS共享存储池和精简配置
- swper 实现滑动切换功能的两种方式
- 7-6 程序员买西瓜 (5分)
- 实现简易字符串压缩算法