✏️
ougege
  • README
  • Docs
    • index
    • Articles
      • AI
        • 体验Chrome AI
        • 体验Cloudflare Workers AI
        • 体验deepseek
      • CSS
        • CSS优化-PurgeCSS
        • 实用效果
        • 开发常用样式
      • Deepin
        • deepin20安装mysql
        • deepin使用tensorflow入门机器学习
        • deepin安装cuda和cuDNN
        • deepin安装lamp
        • deepin安装nvidia驱动
        • deepin安装oh my Zsh
        • deepin安装p7zip
        • deepin换源
        • 安装deepin系统后要做的事
      • Docker
        • CI/CD搭建配置
        • deepin搭建docker环境
        • docker安装和使用gitlab
        • docker搭建nginx+php环境
      • Essay
        • IOS申请邓白氏编码
        • Markdown-Mermaid
        • Markdown Use
        • webview白屏的问题查找和修复
        • 前端开发对接问题和解决办法汇总
        • 国务院机构改革方案
        • 国家级智库
        • 实用网站推荐
        • 常用Markdown数学公式语法
        • 强烈推荐前端要安装的vscode扩展
        • 新建销售计划-页面卡死问题分析
        • 海淘入坑手册
        • 竞品研究
        • 足球知识速成
      • Git
        • GitBook安装和常用命令
        • GitKraken免费版本
        • Git安装和配置
        • Git异常处理
        • Git Worktree使用
        • 前端工程化相关的实用git命令
      • JS
        • ESM模块导出方式对比
        • Emoji多端统一处理
        • JS发布订阅模式
        • JS性能优化
        • JS标准内置对象
        • JS链式调用原理
        • Promise介绍和使用
        • Range的使用
        • Vue+Oauth登录实现
        • Vue实现富文本插入Emoji
        • chrome扩展入门
        • es5新特性
        • es6常用特性
        • es常用片段
        • uniapp使用eslint校验代码
        • 与移动端通信
        • 优秀js库moment
        • 使用vue-socketio
        • 实现一个中间件
        • 小程序webview调试
        • 常用snippets
        • 常用正则
        • 常用的设计模式
        • 微信jssdk封装使用
        • 浏览器宏任务和微任务
        • 浏览器的5种Observer
        • 深入理解赋值、浅拷贝、深拷贝
        • 解析vue指令clickoutside源码
        • 键盘事件与KeyBoardWrapper交互
        • 高德地图常用方法封装
        • 高阶函数片段
      • Network
        • 使用Lighthouse分析前端性能
        • 前后端启用https
        • 宝塔nginx安装ngx_brotli
        • 比较gz与br加载速度
        • 浏览器https提示不安全
        • 浏览器提示HSTS
        • 简单使用tcpdump
        • 静态资源gzip优化
      • Node
        • CommonJS模块导出方式对比
        • Taro command not found 多平台解决方案
        • koa使用和API实现
        • node安装报错Unexpected-token
        • 使用nvm和nrm
        • 使用uniapp给小程序添加云函数
        • 使用verdaccio搭建本地npm仓库
        • 使用vue-cli搭建vue项目
        • 安装Node.js和npm配置
        • 编译成cjs和mjs的思路解析
        • 让你的npmPackage同时支持cjs和mjs
        • 通过GithubAction将内容部署到vps
      • Python
        • Python源管理
        • Python版本管理
        • mitmproxy抓包
        • 微信公众平台开发爬坑经历
      • Shell
        • Ubuntu安装deepin桌面环境
        • Ubuntu安装flatpak软件
        • Ubuntu安装wireshark
        • Ubuntu常见问题汇总
        • dell G3装系统无法识别第二块硬盘
        • linux下virtualbox用gho还原系统
        • mysql常用命令
        • navicat连接一键集成环境的mysql
        • nginx常用命令
        • pm2常用命令
        • virtualbox虚拟机和宿主机互相复制粘贴
        • vps内资源通过mega快传到本地
        • vps报错temporary failure in name resolution
        • vscode修改文件监控数
        • windows+linux双系统引导修复
        • zsh常用插件和命令
        • 一键搭建ChatGPT web版
        • 使用V2ray,CloudFlare Warp解锁GPT
        • 使用vscode进行java开发
        • 利用zx和SSHKey发布代码到服务器
        • 反爬虫一些方案总结和尝试
        • 安装1Panel
        • 安装Bt面板
        • 安装Ubuntu22.04后要做的事
        • 无显示器linux设置默认分辨率
        • 特别实用的shell命令
        • 解决linux安装xmind缺少依赖报错
      • Standards
        • CSS格式化之stylelint
        • CSS规范
        • HTML规范
        • JS规范
        • commit规范
        • 使用husky+commitlint规范代码提交
        • 使用semantic-release自动管理版本号
        • 命名规范
        • 图片规范
        • 版本编号规范
      • Wall
        • 科学上网-Cloudflare-Pages
        • 科学上网-Cloudflare-Warp
        • 科学上网-Geph
        • 科学上网-RackNerd
        • 科学上网-Slicehosting
        • 科学上网-Surfshark
        • 科学上网-Tor
        • 科学上网-XX-NET
        • 科学上网-heroku
        • 科学上网-shadowsock
        • 科学上网-v2ray使用
        • 科学上网-v2ray搭建
        • 科学上网-浏览器代理
        • 科学上网-让终端走代理
      • Windows
        • SourceTree破解免登录(windows版)
        • git bash交互提示符不工作
        • nexus 7 2013 wifi 刷机
        • tree命令生成文件目录
        • 利用charles抓包app
        • 安装Openssl
        • 安装msi文件报错2503和2502
        • 神器vimium使用说明
        • 自用host
        • 解决win10扩展出来的屏幕模糊
        • 解决安装Adobe Air时发生错误
    • Snippets
      • zsh
        • docker
        • extract
        • git-commit
        • git
        • mysql-macports
        • npm
        • nvm
        • pip
        • pm2
        • systemd
        • ubuntu
        • vscode
Powered by GitBook
On this page
  • 介绍
  • 初步判断
  • 顺藤摸瓜
  • 出问题的代码分析
  • 解决方案

Was this helpful?

  1. Docs
  2. Articles
  3. Essay

webview白屏的问题查找和修复

PreviousMarkdown UseNext前端开发对接问题和解决办法汇总

Last updated 1 month ago

Was this helpful?

介绍

有运营反应,有部分用户访问我们的小程序后,打开 webview 页面,显示白屏,删除小程序重新进入也未解决问题。

初步判断

这里有俩个关键点

  1. 仅部分用户出现白屏

  2. 以前是能正常访问

初步判断是代码执行的兼容性问题

顺藤摸瓜

刚巧团队内部有个同事的 iphone 7 能复现这个问题, 于是我们首先进行了 charles 抓包, 分别拿到了正常手机和白屏手机的请求, 对比发现俩者从服务端拿到的资源请求包括文件名 hash 是一致的。

webview_charles_1

到了这里我们基本可以确定不同设备上拿到的代码是一致的,出问题的是兼容性,在不同机型上渲染执行可能出错导致的白屏。

前面提到用户以前是能正常访问 webview 的,也就是在某一段时间我们的代码编译后导致的白屏.

于是我接下来的重点放到了查找问题代码,通过切换月份分支,从 5,6,7 月分别调试缩小到7月范围,然后按照 commit 减半调试缩小范围,确定了包含问题代码的 commit 记录

在多次注释更新的包后,最终确定了问题定位出在 package.json 升级的新包 @muyi086/var-type 上.

出问题的代码分析

  1. 这是老版本的代码,使用 js 实现

    ::: code-group

    /**
     * @Description: js变量类型判断
      * @Author: MuYi086
      * @Email: 1258947325@qq.com
      * @Blog: https://github.com/MuYi086/blog
      * @Date: 2021/04/11 12:30
      */
    class VarType {
      constructor () {
        this.typeList = ['Null', 'Undefined', 'Object', 'Array', 'ArrayBuffer', 'String', 'Number', 'Boolean', 'Function', 'RegExp', 'Date', 'FormData', 'File', 'Blob', 'URLSearchParams', 'Set', 'WeakSet', 'Map', 'WeakMap']
        this.init()
      }
    
      /**
       * 判断变量类型
      * @param {string} value
      * @returns lowercase string
      */
      type (value) {
        const s = Object.prototype.toString.call(value)
        return s.match(/\[object (.*?)\]/)[1].toLowerCase()
      }
    
      /**
       * 增加判断类型数据方法
      */
      init () {
        this.typeList.forEach((t) => {
          this['is' + t] = (o) => {
            return this.type(o) === t.toLowerCase()
          }
        })
      }
    
      /**
       * isBuffer
      * @param {any} val
      * @returns boolean
      */
      isBuffer (val) {
        return val !== null && !this.isUndefined(val) && val.constructor !== null && !this.isUndefined(val.constructor) && this.isFunction(val.constructor.isBuffer) && val.constructor.isBuffer(val)
      }
    
      /**
       * isStream
      * @param {any} val
      * @returns boolean
      */
      isStream (val) {
        return this.isObject(val) && this.isFunction(val.pipe)
      }
    }
    // 使用 varType["isNull"](null)等
    module.exports = new VarType()
    var e = class {
      constructor() {
        this.typeList = ["Null", "Undefined", "Object", "Array", "ArrayBuffer", "String", "Number", "Boolean", "Function", "RegExp", "Date", "FormData", "File", "Blob", "URLSearchParams", "Set", "WeakSet", "Map", "WeakMap"],
        this.init()
      }
      type(t) {
        return Object.prototype.toString.call(t).match(/\[object (.*?)\]/)[1].toLowerCase()
      }
      init() {
        this.typeList.forEach(t = >{
          this["is" + t] = r = >this.type(r) === t.toLowerCase()
        })
      }
      isBuffer(t) {
        return t !== null && !this.isUndefined(t) && t.constructor !== null && !this.isUndefined(t.constructor) && this.isFunction(t.constructor.isBuffer) && t.constructor.isBuffer(t)
      }
      isStream(t) {
        return this.isObject(t) && this.isFunction(t.pipe)
      }
    };
    module.exports = new e;

    :::

  2. 这是升级后新版本的代码,使用 ts 实现

    ::: code-group

    /**
     * @Description: js变量类型判断
      * @Author: MuYi086
      * @Email: 1258947325@qq.com
      * @Blog: https://github.com/MuYi086/blog
      * @Date: 2021/04/11 12:30
      */
    class VarType {
      private typeList: string[]
      private static _instance: VarType | null = null
      constructor() {  
        this.typeList = ['Null', 'Undefined', 'Object', 'Array', 'ArrayBuffer', 'String', 'Number', 'Boolean', 'Function', 'RegExp', 'Date', 'FormData', 'File', 'Blob', 'URLSearchParams', 'Set', 'WeakSet', 'Map', 'WeakMap']
        this.init()
      }
      static get instance(): VarType {  
        if (!VarType._instance) {
          VarType._instance = new VarType()
        }
        return VarType._instance
      }
      /**
       * 判断变量类型
      * @param {string} value
      * @returns lowercase string
      */
      private type (value: any): string {
        const s = Object.prototype.toString.call(value)
        return s.match(/\[object (.*?)\]/)[1].toLowerCase()
      }
    
      /**
       * 增加判断类型数据方法
      */
      private init(): void {
        const self = this
        this.typeList.forEach((t: string) => {
          Object.defineProperty(VarType.prototype, `is${t}`, {
            value: function (o: any) {
              return self.type(o) === t.toLowerCase()
            },
            writable: true,
            configurable: true
          })
        })
      }
      /**
       * isBuffer
      * @param {any} val
      * @returns boolean
      */
      static isBuffer(val: any): boolean {  
        return val !== null && (VarType as any).isUndefined(val) && val.constructor !== null && (VarType as any).isUndefined(val.constructor) && typeof val.constructor.isBuffer === 'function' && val.constructor.isBuffer(val)
      }
    
      /**
       * isStream
      * @param {any} val
      * @returns boolean
      */
      static isStream(val: any): boolean {  
        return (VarType as any).isObject(val) && (VarType as any).isFunction(val.pipe)
      }
    }
    // 使用 varType["isNull"](null)等
    export const varType = VarType.instance
    var s = Object.defineProperty;
    var o = Object.getOwnPropertyDescriptor;
    var c = Object.getOwnPropertyNames;
    var u = Object.prototype.hasOwnProperty;
    var p = (e, t) = >{
      for (var n in t) s(e, n, {
        get: t[n],
        enumerable: !0
      })
    },
    y = (e, t, n, i) = >{
      if (t && typeof t == "object" || typeof t == "function") for (let r of c(t)) ! u.call(e, r) && r !== n && s(e, r, {
        get: () = >t[r],
        enumerable: !(i = o(t, r)) || i.enumerable
      });
      return e
    };
    var f = e = >y(s({},
    "__esModule", {
      value: !0
    }), e);
    var b = {};
    p(b, {
      varType: () = >l
    });
    module.exports = f(b);
    var a = class e {
      typeList;
      static _instance = null;
      constructor() {
        this.typeList = ["Null", "Undefined", "Object", "Array", "ArrayBuffer", "String", "Number", "Boolean", "Function", "RegExp", "Date", "FormData", "File", "Blob", "URLSearchParams", "Set", "WeakSet", "Map", "WeakMap"],
        this.init()
      }
      static get instance() {
        return e._instance || (e._instance = new e),
        e._instance
      }
      type(t) {
        return Object.prototype.toString.call(t).match(/\[object (.*?)\]/)[1].toLowerCase()
      }
      init() {
        let t = this;
        this.typeList.forEach(n = >{
          Object.defineProperty(e.prototype, `is$ {
            n
          }`, {
            value: function(i) {
              return t.type(i) === n.toLowerCase()
            },
            writable: !0,
            configurable: !0
          })
        })
      }
      static isBuffer(t) {
        return t !== null && e.isUndefined(t) && t.constructor !== null && e.isUndefined(t.constructor) && typeof t.constructor.isBuffer == "function" && t.constructor.isBuffer(t)
      }
      static isStream(t) {
        return e.isObject(t) && e.isFunction(t.pipe)
      }
    },
    l = a.instance;

    :::

对比发现 ts 的代码实现使用了单例模式,并且内部使用了defineProperty, getOwnPropertyDescriptor 等属性,这也是导致某些机型上不执行代码导致白屏的原因, 因为在编译时没有提供 polyfill 支持旧版浏览器。

解决方案

由于线上 bug 需要紧急修复,所以当前先简单处理,发布一个新的 npm 包 js 版本替换原先的 ts 版本, 然后紧急发版。以后有时间会优化代码,在 ts 版本上尝试修复编译导致的问题。

webview_commit_1
webview_commit_2