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

相关推荐
爱学习的程序媛20 小时前
【Web前端】JavaScript设计模式全解析
前端·javascript·设计模式·web
薛先生_09921 小时前
js学习语法第一天
开发语言·javascript·学习
苦瓜小生21 小时前
【前端】|【js手撕】经典高频面试题:手写实现function.call、apply、bind
java·前端·javascript
踩着两条虫21 小时前
AI驱动的Vue3应用开发平台深入探究(十):物料系统之内置组件库
android·前端·vue.js·人工智能·低代码·系统架构·rxjava
和沐阳学逆向21 小时前
我现在怎么用 CC Switch 管中转站,顺手拿 Codex 举个例子
开发语言·javascript·ecmascript
慧一居士1 天前
nuxt3 项目和nuxt4 项目区别和对比
前端·vue.js
kgduu1 天前
js之客户端存储
javascript·数据库·oracle
四千岁1 天前
2026 最新版:WSL + Ubuntu 全栈开发环境,一篇搞定!
javascript·node.js
竹林8181 天前
从“连接失败”到丝滑登录:我用 ethers.js 连接 MetaMask 的完整踩坑实录
前端·javascript
神舟之光1 天前
jwt权限控制简单总结(乡村意见簿-vue-express-mongdb)
前端·vue.js·express