踩坑日记 Vue Cannot read properties of undefined (reading '_wrapper')

前言

最近遇到一个比较紧急的需求,由于时间紧张需要同步大量的代码,因此我开始了频繁的复制粘贴,删除修改代码的工作。

在我处理一个弹窗相关的问题时,我遇到了比较有意思的异常:

js 复制代码
 Cannot read properties of undefined (reading '_wrapper')

问题排查

当时的控制台报错类似如下:

明眼人一下子就注意到了,上面xxx 函数未定义,click 事件拿到一个 undefined

当时的情况比较复杂,控制台的异常远远不止这些,而且高强度的开发让我没注意到上面的提示信息。

于是直接从这个报错下手:

js 复制代码
 Cannot read properties of undefined (reading '_wrapper')

(1)确认异常代码

我们可以看到 _wrapper 是一个没有被定义的变量,于是我先从异常提示中进行排查,点开异常之后我们可以看到下面的代码提示:

显然这并不是我的代码,应该是某些依赖的内容报错了,我首先怀疑的是这个vue的源码。

(2)查看异常代码

打开本地的vue源码,很久没复习的缘故很多细节记不清了,但是大概流程还是知道的。

我们直接在vue源码项目中搜索 _wrapper,我们可以看到代码如下所示:

代码解释

  • add 函数用于向目标元素添加事件监听器。
  • 它接受四个参数: name 表示事件名称, handler 表示事件处理函数, capture 表示是否在捕获阶段触发事件, passive 表示是否以被动方式监听事件。
  • 在函数内部,它首先处理了一个异步边缘情况,即当点击事件触发补丁(patch)时,事件处理程序附加到外部元素并再次触发。
  • 函数保存了事件处理程序附加的时间戳,并且只有当传递给它的事件的时间戳晚于附加时间戳时,事件处理程序才会触发。
  • 最后,函数通过调用 target.addEventListener 方法将事件监听器添加到目标元素上。
  • 如果浏览器支持 passive 选项,则会将 { capture, passive } 作为第三个参数传递给 addEventListener 方法,否则只传递 capture 参数。

(3)错误原因

核心如下:

js 复制代码
const original = handler;
handler = original._wrapper = function (e) {
      if (
        // no bubbling, should always fire.
        // this is just a safety net in case event.timeStamp is unreliable in
        // certain weird environments...
        e.target === e.currentTarget ||
        // event is fired after handler attachment
        e.timeStamp >= attachedTimestamp ||
        // bail for environments that have buggy event.timeStamp implementations
        // #9462 iOS 9 bug: event.timeStamp is 0 after history.pushState
        // #9681 QtWebEngine event.timeStamp is negative value
        e.timeStamp <= 0 ||
        // #9448 bail if event is fired in another document in a multi-page
        // electron/nw.js app, since event.timeStamp will be using a different
        // starting reference
        e.target.ownerDocument !== document
      ) {
        return original.apply(this, arguments)
      }
    }

代码接受了handler 事件处理函数,赋值给 original,然后original._wrapper 赋值给 handler

假如 handler 为undefined,那么 original._wrapper 赋值给 handler 就会出现 _wrapper未定义错误,同时也会提示绑定事件got 一个 undefined的内容,所以问题原因就是存在未注册的函数。

问题处理

排除控制台中提示的未定义函数名称,并移除这个函数名称相关的事件绑定即可。

总结

这个问题很简单,主要还是我不够细心导致的。

但在这个问题中也可以了解到,对于频繁开发的vue或者react项目的同学来说,学习源码还是很重要的,必要时面向源码开发可以解决很多问题。

相关推荐
JNU freshman1 天前
vue 之 import 的语法
前端·javascript·vue.js
剑亦未配妥1 天前
Vue 2 响应式系统常见问题与解决方案(包含_demo以下划线开头命名的变量导致响应式丢失问题)
前端·javascript·vue.js
爱吃的强哥1 天前
Vue2 封装二维码弹窗组件
javascript·vue.js
凉柚ˇ1 天前
Vue图片压缩方案
前端·javascript·vue.js
ByteCraze1 天前
秋招被问到的常见问题
开发语言·javascript·原型模式
优弧1 天前
Vue 和 React 框架对比分析:优缺点与使用场景
vue.js
渣哥1 天前
从代理到切面:Spring AOP 的本质与应用场景解析
javascript·后端·面试
UIUV1 天前
JavaScript代理模式实战解析:从对象字面量到情感传递的优雅设计
javascript
Kimser1 天前
基于 VxeTable 的高级表格选择组件
前端·vue.js