it编程 > 编程语言 > Asp.net

WebView渲染异常导致闪退问题的解决方案

19人参与 2025-02-27 Asp.net

背景:

app主页面使用了大量webview容器(10个以上)显示图表信息,最新发现bugly上面出现一些关于浏览器native crash,如下:

经排查,是webview渲染失败导致crash,可以通过webview.loadurl("chrome://crash")模拟。

解决方法:

1、通过设置webviewclient,重写onrenderprocessgone()返回值,强制返回true,表示在webview发生异常时,自己处理,这样app就不会出现crash。这么做app虽然没有crash,但是主页面的webview内容却看不到了,看到的是白色/黑色背景,体验极差。

2、要想解决webview内容不可见问题,还需要在web出现异常的时候,移除原有web容器,重新创建一个web容器,代码如下:

class reportwebview @jvmoverloads constructor(context: context, attrs: attributeset? = null) : wendu.dsbridge.basewebview(context, attrs) {

    var reloadfun: ((any: reportwebview) -> unit)? = null
    private var parentviewgroup: viewgroup? = null

    init {
        webviewclient = customwebviewclient()
    }

    override fun onattachedtowindow() {
        super.onattachedtowindow()
        parentviewgroup = parent as? viewgroup
    }

    private inner class customwebviewclient : webviewclient() {

        override fun onrenderprocessgone(view: webview?, detail: renderprocessgonedetail?): boolean {
            // 所有的web都crash,所以都需要重建
            recreatewebviewandreload(view)
            return true
        }
    }

    private fun recreatewebviewandreload(view: webview?) {
        val originalurl = view?.url// 原始webview地址
        val isvisible = view?.isvisible
        val lp = this.layoutparams
        // 移除旧的 webview
        val index = indexinparent()
        if (parentviewgroup != null) {
            parentviewgroup?.removeview(this)
        }
        destroy()// 销毁

        // 重新创建 webview
        val newwebview = reportwebview(context)
        reloadfun?.invoke(newwebview)
        newwebview.reloadfun = reloadfun
        newwebview.id = id
        newwebview.layoutparams = lp
        newwebview.isvisible = isvisible.nullor(false)
        originalurl?.let { newwebview.loadurl(it) }

        // 将新的 webview 添加回布局中
        parentviewgroup?.addview(newwebview, index)

        // 更新引用
        parentviewgroup = newwebview.parent as? viewgroup
    }

    private fun indexinparent(): int {
        return parentviewgroup?.indexofchild(this) ?: -1
    }

}

本项目桥接使用的是dsbridge三方库,在创建web容器需要设置addjavascriptobject(),即reloadfun函数。

注意事项:

1、同一个页面只要有一个渲染异常,会导致所有web容器异常,所以所有web容器都要重新创建,不可以根据web可见状态只创建可见的web。

2、在使用的时候,原有的web容器已被移除,需要使用最新的web容器,否则就会报错。上述代码中,新的web容器id跟移除的一样,所以也很容易拿到新的web容器,代码如下:

/**
 * 获取真实的webview,之前的web可能被销毁
 * @param id web id
 */
private fun getrealwebview(id: int): reportwebview {
    return mbinding.root.findviewbyid(id)
}

总结 

到此这篇关于webview渲染异常导致闪退问题解决方案的文章就介绍到这了,更多相关webview渲染异常闪退内容请搜索代码网以前的文章或继续浏览下面的相关文章希望大家以后多多支持代码网!

(0)
打赏 微信扫一扫 微信扫一扫

您想发表意见!!点此发布评论

推荐阅读

C# swagger ui增加访问限制方式

02-27

C#实现tostring转换成16进制的方法

02-27

C#项目中引用Swagger的详细步骤和配置方式

02-27

C# string转unicode字符的实现

02-26

C#使用Linq实现简单去重处理

02-20

C#集成DeepSeek模型实现AI私有化的流程步骤(本地部署与API调用教程)

03-03

猜你喜欢

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论