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

相关推荐
赵大仁18 分钟前
React Native 与 Expo
javascript·react native·react.js
程序员与背包客_CoderZ1 小时前
Node.js异步编程——Callback回调函数实现
前端·javascript·node.js·web
清灵xmf2 小时前
从 Set、Map 到 WeakSet、WeakMap 的进阶之旅
前端·javascript·set·map·weakset·weakmap
运维@小兵3 小时前
vue使用路由技术实现登录成功后跳转到首页
前端·javascript·vue.js
肠胃炎3 小时前
React构建组件
前端·javascript·react.js
邝邝邝邝丹3 小时前
React学习———React.memo、useMemo和useCallback
javascript·学习·react.js
美酒没故事°3 小时前
纯css实现蜂窝效果
前端·javascript·css
GISer_Jing4 小时前
React useState 的同步/异步行为及设计原理解析
前端·javascript·react.js
mini榴莲炸弹4 小时前
什么是SparkONYarn模式?
前端·javascript·ajax
能来帮帮蒟蒻吗4 小时前
VUE3 -综合实践(Mock+Axios+ElementPlus)
前端·javascript·vue.js·笔记·学习·ajax·typescript