Android动态实例化view,Android旁门左道之动态替换系统View类

2024-02-24 11:30

本文主要是介绍Android动态实例化view,Android旁门左道之动态替换系统View类,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

66b52468c121889b900d4956032f1009.png

8种机械键盘轴体对比

本人程序员,要买一个写代码的键盘,请问红轴和茶轴怎么选?

导语 本文讲述如何通过替换系统View类的方法,定位一个特殊机型问题

作者: yarkeyzhang  2017.6.29

一,ImageView抛来一个异常

应用程序Crash是Android

App开发习以为常的问题,大部分Crash我们通过日志找到调用栈可以很快定位到出错的代码。然而有一些Crash却显得没那么直接,比如下面这个由Android系统抛(throw)出来的异常。

17-06-06 11:36:20|1496720179572[20047]1|E|StatisticCollector|

getCrashExtraMessage...

isNativeCrashed: false

crashType=java.lang.RuntimeException

crashAddress=android.graphics.Canvas.throwIfCannotDraw(Canvas.java:1270)

crashStack=android.graphics.Canvas.throwIfCannotDraw(Canvas.java:1270)

android.graphics.Canvas.drawBitmap(Canvas.java:1404)

android.graphics.drawable.BitmapDrawable.draw(BitmapDrawable.java:544)

android.widget.ImageView.onDraw(ImageView.java:1246)

android.view.View.draw(View.java:16245)

android.view.View.updateDisplayListIfDirty(View.java:15242)

android.view.View.draw(View.java:16015)

android.view.ViewGroup.drawChild(ViewGroup.java:3740)

android.view.ViewGroup.dispatchDraw(ViewGroup.java:3530)

android.view.View.draw(View.java:16248)

android.view.View.updateDisplayListIfDirty(View.java:15242)

android.view.View.draw(View.java:16015)

android.view.ViewGroup.drawChild(ViewGroup.java:3740)

android.view.ViewGroup.dispatchDraw(ViewGroup.java:3530)

android.view.View.updateDisplayListIfDirty(View.java:15237)

android.view.View.draw(View.java:16015)

android.view.ViewGroup.drawChild(ViewGroup.java:3740)

android.view.ViewGroup.dispatchDraw(ViewGroup.java:3530)

android.view.View.updateDisplayListIfDirty(View.java:15237)

android.view.View.draw(View.java:16015)

android.view.ViewGroup.drawChild(ViewGroup.java:3740)

android.view.ViewGroup.dispatchDraw(ViewGroup.java:3530)

android.view.View.draw(View.java:16248)

com.android.internal.policy.PhoneWindow$DecorView.draw(PhoneWindow.java:2822)

android.view.View.updateDisplayListIfDirty(View.java:15242)

android.view.ThreadedRenderer.updateViewTreeDisplayList(ThreadedRenderer.java:282)

android.view.ThreadedRenderer.updateRootDisplayList(ThreadedRenderer.java:288)

android.view.ThreadedRenderer.draw(ThreadedRenderer.java:323)

android.view.ViewRootImpl.draw(ViewRootImpl.java:2649)

android.view.ViewRootImpl.performDraw(ViewRootImpl.java:2468)

android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:2072)

android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1108)

android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:6146)

android.view.Choreographer$CallbackRecord.run(Choreographer.java:892)

android.view.Choreographer.doCallbacks(Choreographer.java:704)

android.view.Choreographer.doFrame(Choreographer.java:640)

android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:878)

android.os.Handler.handleCallback(Handler.java:739)

android.os.Handler.dispatchMessage(Handler.java:95)

android.os.Looper.loop(Looper.java:148)

android.app.ActivityThread.main(ActivityThread.java:5648)

我们看看日志,ViewRootImpl.doTraversal()是遍历Window所有View刷新界面的过程,这个过程由系统触发,我们在调用栈中找不到任何的App客户代码。过程中,ImageView在执行onDraw()的时候出现了异常。这是某部手机在开启多窗口模式时必现Crash。怎么办?

二、寻求解决思路

这个问题出现在一个编辑图片的页面,页面中含有很多的ImageView(大约20个)实例,单单靠调用栈我们无法定位具体哪个ImageView出现了问题。

不过相信大家已经有很多解决思路:

1. 通过日志文件寻找出错前后是否有更多帮助信息,配合源码定位问题

2. 借到问题手机,连接电脑配合源码打断点(ImageView,BitmapDrawable,Canvas)

思路1无法快速解决问题;思路2恕我实在借不到那个型号的手机,另外我们IDE中的Android源码与手机中行数不一定匹配,给ImageView,BitmapDrawable等等这些系统类打断点,代码行数对不上的话也就很难搞。

这里我想到了一个思路:能不能重写ImageView.onDraw()方法,在出现异常时打印出所有我们需要的日志信息(比如view id)

三、往LayoutInflater下手

重写ImageView.onDraw()方法实际上等于我们需要替换ImageView类,把所有的xml布局文件中的ImageView换成我们新定义的CatchExceptionImageView?这个显然不太好办。最后我在LayoutInflater类中找到了方法。

ad947170e91ba0b5e64f376019da94b3552102cf14d1ac876cbced9cbb5c6873

每个Activity拥有一个LayoutInflater 对象,它负责解析Android xml 布局文件然后实例化View或者View子类对象。核心函数是

LayoutInflater.createViewFromTag(View parent, String name, AttributeSet

attrs),它通过xml标签指定的类名字,实例化出View对象。在这里做手脚,我们可以将xml中所有的标签实例化成

CatchExceptionImageView。

950108b9e6717511666023bd26ec644fc640244f5123b63164ee9dbe12059260

查看createViewFromTag()源码我们可以发现, LayoutInflater其实支持外部提供工厂类来自定义View的创建机制,对应的方法是

setFactory() 和 setFactory2()。

如果大家有用过android.support.v7.app.AppCompatActivity,那么你会发现,xml布局中的Button标签实际上创建了android.support.v7.widget.AppCompatButton对象,TextView标签实际上创建了android.support.v7.widget.AppCompatTextView对象,这是通过LayoutInflater.Factory来影响View的创建实现的。

所以,我们调用 setFactory()或者setFactory2()方法有可能会遇到失败:“A factory has already been set

on this LayoutInflater”。最后,我通过反射把我定义的Factory对象安全地注入到了LayoutInflater对象中。

四、调试代码协助定位问题

为了捕捉到抛出异常的ImageView,我大概写了下面这样的代码:

public static class CatchExceptionImageView extends ImageView {

public CatchExceptionImageView(Context context) {

super(context);

}

public CatchExceptionImageView(Context context, AttributeSet attrs) {

super(context, attrs);

}

@Override

protected void onDraw(Canvas canvas) {

Drawable drawable = getDrawable();

if (drawable instanceof BitmapDrawable) {

BitmapDrawable bitmapDrawable = (BitmapDrawable) drawable;

if (bitmapDrawable.getBitmap().isRecycled()) {

Log.e(TAG, "we'll crash !! " + this);

}

}

super.onDraw(canvas);

}

}

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

HashMap hookViewMap = new HashMap();

hookViewMap.put("ImageView", CatchExceptionImageView.class.getName());

new LayoutDebugHelper().onActivityCreate(this, hookViewMap);

setContentView(R.layout.activity_main);

}

我构建了新的包发送给远方的优测测试人员,帮我复现了问题并抓了日志,最后找到了Crash的ImageView信息,通过view id便可以找到了出错的点。

够折腾吧,哈哈,机型问题虐我千百遍!

全文完,感谢阅读!

这篇关于Android动态实例化view,Android旁门左道之动态替换系统View类的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/741957

相关文章

vue基于ElementUI动态设置表格高度的3种方法

《vue基于ElementUI动态设置表格高度的3种方法》ElementUI+vue动态设置表格高度的几种方法,抛砖引玉,还有其它方法动态设置表格高度,大家可以开动脑筋... 方法一、css + js的形式这个方法需要在表格外层设置一个div,原理是将表格的高度设置成外层div的高度,所以外层的div需要

C#实现添加/替换/提取或删除Excel中的图片

《C#实现添加/替换/提取或删除Excel中的图片》在Excel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更加美观,下面我们来看看如何在C#中实现添加/替换/提取或删除E... 在Excandroidel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更

C#实现系统信息监控与获取功能

《C#实现系统信息监控与获取功能》在C#开发的众多应用场景中,获取系统信息以及监控用户操作有着广泛的用途,比如在系统性能优化工具中,需要实时读取CPU、GPU资源信息,本文将详细介绍如何使用C#来实现... 目录前言一、C# 监控键盘1. 原理与实现思路2. 代码实现二、读取 CPU、GPU 资源信息1.

SpringBoot实现动态插拔的AOP的完整案例

《SpringBoot实现动态插拔的AOP的完整案例》在现代软件开发中,面向切面编程(AOP)是一种非常重要的技术,能够有效实现日志记录、安全控制、性能监控等横切关注点的分离,在传统的AOP实现中,切... 目录引言一、AOP 概述1.1 什么是 AOP1.2 AOP 的典型应用场景1.3 为什么需要动态插

java图像识别工具类(ImageRecognitionUtils)使用实例详解

《java图像识别工具类(ImageRecognitionUtils)使用实例详解》:本文主要介绍如何在Java中使用OpenCV进行图像识别,包括图像加载、预处理、分类、人脸检测和特征提取等步骤... 目录前言1. 图像识别的背景与作用2. 设计目标3. 项目依赖4. 设计与实现 ImageRecogni

在C#中获取端口号与系统信息的高效实践

《在C#中获取端口号与系统信息的高效实践》在现代软件开发中,尤其是系统管理、运维、监控和性能优化等场景中,了解计算机硬件和网络的状态至关重要,C#作为一种广泛应用的编程语言,提供了丰富的API来帮助开... 目录引言1. 获取端口号信息1.1 获取活动的 TCP 和 UDP 连接说明:应用场景:2. 获取硬

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

2.1/5.1和7.1声道系统有什么区别? 音频声道的专业知识科普

《2.1/5.1和7.1声道系统有什么区别?音频声道的专业知识科普》当设置环绕声系统时,会遇到2.1、5.1、7.1、7.1.2、9.1等数字,当一遍又一遍地看到它们时,可能想知道它们是什... 想要把智能电视自带的音响升级成专业级的家庭影院系统吗?那么你将面临一个重要的选择——使用 2.1、5.1 还是

Java操作ElasticSearch的实例详解

《Java操作ElasticSearch的实例详解》Elasticsearch是一个分布式的搜索和分析引擎,广泛用于全文搜索、日志分析等场景,本文将介绍如何在Java应用中使用Elastics... 目录简介环境准备1. 安装 Elasticsearch2. 添加依赖连接 Elasticsearch1. 创

使用C#代码计算数学表达式实例

《使用C#代码计算数学表达式实例》这段文字主要讲述了如何使用C#语言来计算数学表达式,该程序通过使用Dictionary保存变量,定义了运算符优先级,并实现了EvaluateExpression方法来... 目录C#代码计算数学表达式该方法很长,因此我将分段描述下面的代码片段显示了下一步以下代码显示该方法如