当前位置: 首页 > news >正文

诡异!React stopPropagation失灵

前言

先来看这
故事的开头是这样的:写代码的过程中,发现在内层用stopPropagation阻止绑定在document上的事件的时候,是没办法做到的,只可以阻止outClick事件的触发。

class ExampleApplication extends React.Component {
    componentDidMount() {
        document.addEventListener('click', () => {
            alert('document click');
        })
    }

    outClick(e) {
        console.log(e.currentTarget);
        alert('outClick');
    }
    
    onClick(e) {
        console.log(e.currentTarget);
        alert('onClick');
        e.stopPropagation();
    }
    render() {
        return <div onClick={this.outClick}>
            <button onClick={this.onClick}> 测试click事件 </button>
        </div>
    }
}

关于这个问题的解释,网上五花八门,有些将其归结为是由于事件委托的原因,例如这篇文章里说

我们直接在jsx模板上绑定的事件,都是委托在了document上,那自然要比直接在dom上绑定的事件慢了,等document收到事件后才去e.stopPropagation(),太晚了

其实从上面例子的输出: 由'onClick',再到'document click'可知,其实原生document上绑定的事件时最后执行的,所以并不是因为document收到事件快慢的原因而导致这个问题。

真相

真相只有一个,那就是:
出现上述bug的主要原因是混用浏览器原生事件跟React合成事件

详细解释:React有自己的一套事件处理机制,它会将所有的事件都绑定在document上,然后再用dispatchEvent统一分发,这时候分发的是合成事件
onClick(e)这时候拿到的e其实是合成事件,只能阻止合成事件的冒泡。

举个栗子来验证一下


class ExampleApplication extends React.Component {
    componentDidMount() {
        document.addEventListener('click', () => {
            alert('document click');
        })
        document.getElementById('div1').addEventListener('click', () => {
            alert('原生outClick');
        })
    }

    outClick(e) {
        console.log(e.currentTarget);
        alert('合成outClick');
    }
    
    onClick(e) {
        console.log(e.currentTarget);
        alert('onClick');
        e.stopPropagation();
    }
    render() {
        return <div id="div1" onClick={this.outClick}>
            <button onClick={this.onClick}> 测试click事件 </button>
        </div>
    }
}

做了点小改动,就是外层的div绑定的函数用原生的方式跟jsx的方式都绑定一次,所以最终的输出为

'原生outClick', 'onClick','document click'

所以button回调函数里的stopPropagation只能阻止合成事件的冒泡,而对于原生绑定的,则不行。

解决方法

解决方法有几种,我个人认为最简单的就是直接在onClick里利用[event.stopImmediatePropagation()][2]。原理是这样的:

对于例子一里,document其实绑定了两个事件:

// react 合成事件, dispatchEvent里面执行回调函数
document.addEventListener('click', dispatchEvent);

// 浏览器原生
document.addEventListener('click', () => {
   alert('document click');
})

dispatchEvent里的stopImmediatePropagation可以使得绑定在document上的其他事件就不会被触发

    onClick(e) {
        console.log(e.currentTarget);
        alert('onClick');
        e.nativeEvent.stopImmediatePropagation();
    }

当然本篇文章只是为了抛砖引玉,重点是下一篇的react事件机制,欢迎继续收看。

相关文章:

  • Unix环境高级编程(四)数据系统文件和信息
  • 算法初级之二
  • iOS仿今日头条、壁纸应用、筛选分类、三方微博、颜色填充等源码
  • 人民日报发声,区块链成“兵家必争之地”,或成“国家战略”
  • iOS下JS与OC互相调用(八)--Cordova详解+实战
  • itextsharp display:none无效的bug
  • SaaS型平台产品的POS营销经营策略
  • BeanUtils工具类的使用
  • Linux配置虚拟网站主机
  • 阿里云ECS服务器安装jdk
  • 素材哪里找?一个好的图库
  • POJ1631 LIS模板
  • pyqt5 QGraphicsView颜色动画问题(不兼容,运行不了动画)
  • Java Eclipse和MyEclipse快捷键
  • linux 使用fdisk分区扩容,看介绍命令(未完)
  • “寒冬”下的金三银四跳槽季来了,帮你客观分析一下局面
  • 【347天】每日项目总结系列085(2018.01.18)
  • 【5+】跨webview多页面 触发事件(二)
  • 【每日笔记】【Go学习笔记】2019-01-10 codis proxy处理流程
  • CSS居中完全指南——构建CSS居中决策树
  • eclipse的离线汉化
  • Facebook AccountKit 接入的坑点
  • golang中接口赋值与方法集
  • java小心机(3)| 浅析finalize()
  • JS数组方法汇总
  • js数组之filter
  • Mithril.js 入门介绍
  • PAT A1092
  • SegmentFault 2015 Top Rank
  • Web设计流程优化:网页效果图设计新思路
  • Work@Alibaba 阿里巴巴的企业应用构建之路
  • 大快搜索数据爬虫技术实例安装教学篇
  • 大数据与云计算学习:数据分析(二)
  • 分享几个不错的工具
  • 好的网址,关于.net 4.0 ,vs 2010
  • 解决iview多表头动态更改列元素发生的错误
  • 漂亮刷新控件-iOS
  • 软件开发学习的5大技巧,你知道吗?
  • 深入体验bash on windows,在windows上搭建原生的linux开发环境,酷!
  • 实现菜单下拉伸展折叠效果demo
  • 一文看透浏览器架构
  • 智能情侣枕Pillow Talk,倾听彼此的心跳
  • #[Composer学习笔记]Part1:安装composer并通过composer创建一个项目
  • #考研#计算机文化知识1(局域网及网络互联)
  • ()、[]、{}、(())、[[]]命令替换
  • (3)STL算法之搜索
  • (LeetCode 49)Anagrams
  • (NSDate) 时间 (time )比较
  • (一)appium-desktop定位元素原理
  • (原創) 人會胖會瘦,都是自我要求的結果 (日記)
  • (正则)提取页面里的img标签
  • ./mysql.server: 没有那个文件或目录_Linux下安装MySQL出现“ls: /var/lib/mysql/*.pid: 没有那个文件或目录”...
  • .net redis定时_一场由fork引发的超时,让我们重新探讨了Redis的抖动问题
  • .NET/C# 使用 #if 和 Conditional 特性来按条件编译代码的不同原理和适用场景
  • .NET开发不可不知、不可不用的辅助类(三)(报表导出---终结版)