踩坑日记 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项目的同学来说,学习源码还是很重要的,必要时面向源码开发可以解决很多问题。

相关推荐
遂心_5 小时前
JavaScript 函数参数传递机制:一道经典面试题解析
前端·javascript
Gracemark6 小时前
高德地图-地图选择经纬度问题【使用输入提示-使用Autocomplete进行联想输入】(复盘)
vue.js
遂心_6 小时前
深入理解 React Hook:useEffect 完全指南
前端·javascript·react.js
前端Hardy6 小时前
HTML&CSS: 谁懂啊!用代码 “擦去”图片雾气
前端·javascript·css
前端Hardy6 小时前
HTML&CSS:好精致的导航栏
前端·javascript·css
天下无贼7 小时前
【手写组件】 Vue3 + Uniapp 手写一个高颜值日历组件(含跨月补全+今日高亮+选中状态)
前端·vue.js
一个不爱写代码的瘦子7 小时前
迭代器和生成器
前端·javascript
洋葱头_8 小时前
vue3项目不支持低版本的android,如何做兼容
前端·vue.js
奔跑的蜗牛ing8 小时前
Vue3 + Element Plus 输入框省略号插件:零侵入式全局解决方案
vue.js·typescript·前端工程化
ygria8 小时前
样式工程化:如何实现Design System
前端·前端框架·前端工程化