Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 1229|回复: 0

React+Webpack+ES6 兼容低版本浏览器(IE9)解决方案

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-6-13 13:40:14 | 显示全部楼层 |阅读模式

    虽然过了兼容IE6的噩梦时代,IE依旧阴魂不散,因为你可能还要兼容IE9。在ES6已经普及的今天,用ES6写react已经成了标配。但是babel编译的js语法,由于某些不规范的写法,可能在IE9下不能正确解释,很容易导致白屏。本文记录如下

    起因

    在准备提测的那天,顺便打开IE9看一眼(注意,这里是原生IE9 ,不是用IE11模拟的IE9),OMG!

    图片描述

    排查后发现,原来是因为构造函数中使用了this。简写如下

    
    class Child extends React.Component {
      constructor(props) {
        super(props);
        this.state = {count:this.props.count}
      }
    
      render(){
        return (<p>child</p>)
      }
    }
    class Superer extends React.Component {
      state = {count:1}
      render() {
        return <Child count = {this.state.count}/>
      }
    }
    

    老司机们肯定能一眼发现问题:this.state = {count:this.props.count} 构造函数中不应该使用this,而是 super(props)传入的 porps,应该改为this.state = {count:props.count}. 改正之后,问题确实解决了。但是问题来了,虽然写法确实不规范,为什么其他浏览器都运行正常,包括IE11,用IE11模拟iE9也没有问题,偏偏就原版的IE9有问题。

    怎么能就这么不明不白的算了,哼!

    原因

    既然浏览器运行的代码是经过babel编译的,那这个锅先甩给babel。查看一下babel编译后的源码。如下

    "use strict";
    
    var _createClass = function () { 
    function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
    
    function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
    
    function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
    
    function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
    
    var Child = function (_React$Component) {
      _inherits(Child, _React$Component);
    
      function Child(props) {
        _classCallCheck(this, Child);
    
        var _this = _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).call(this, props));
    
        _this.state = { count: _this.props.count };
        return _this;
      }
    
      _createClass(Child, [{
        key: "render",
        value: function render() {
          return React.createElement(
            "p",
            null,
            "child"
          );
        }
      }]);
    
      return Child;
    }(React.Component);
    
    var Superer = function (_React$Component2) {
      _inherits(Superer, _React$Component2);
    
      function Superer() {
        var _ref;
    
        var _temp, _this2, _ret;
    
        _classCallCheck(this, Superer);
    
        for (var _len = arguments.length, args = Array(_len), _key = 0; _key < _len; _key++) {
          args[_key] = arguments[_key];
        }
    
        return _ret = (_temp = (_this2 = _possibleConstructorReturn(this, (_ref = Superer.__proto__ || Object.getPrototypeOf(Superer)).call.apply(_ref, [this].concat(args))), _this2), _this2.state = { count: 1 }, _temp), _possibleConstructorReturn(_this2, _ret);
      }
    
      _createClass(Superer, [{
        key: "render",
        value: function render() {
          return React.createElement(Child, { count: this.state.count });
        }
      }]);
    
      return Superer;
    }(React.Component);

    重点看_inherits()和Child构造函数,

    subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); 
    if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass;
     
    function Child(props) {
      _classCallCheck(this, Child);
      var _this = _possibleConstructorReturn(this, (Child.__proto__ || Object.getPrototypeOf(Child)).call(this, props));
    
      _this.state = { count: _this.props.count };
      return _this;
    }

    找不到的就是 _this.props.cout,显然,_this指向错误了。查阅(谷)资料(歌)后发现,
    getPrototypeOf() 是 ES5 的方法,IE9+ 都能得到很好的支持,而 setPrototypeOf(),subClass.__proto__ = superClass 是 ES6 的方法,需要到 IE11 才支持,所以_this其实指向的是Function.prototype,而不是react.Component。所以props没有成功赋给Child类,当然就找不到了。

    解决方法

    果然这个锅是babel的。

    那要怎么解决呢?如果是自己写的逻辑,直接修改写法就可以了。但是,如果你用了开源组件,看了源码,找到问题,提了issue,开发者还跟你互动,就说没问题,他还说他亲测没问题,就是不改,你该怎么办?(手动微笑脸)
    当然是原(huan)谅(zu)他(jian)啊~~ ,既然锅是babel的,那就肯定还有一种解决方法。

    使用babel插件babel-preset-es2015-ie
    该插件,在检测到setPrototypeOf(),subClass.__proto__不支持时,自己包装了一个方法

    function _inherits(subClass, superClass) { 
      ...; 
      if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : _defaults(subClass, superClass);
    }
    function _defaults(obj, defaults) {
     var keys = Object.getOwnPropertyNames(defaults);
      for (var i = 0; i < keys.length; i++) {
       var key = keys; 
       var value = Object.getOwnPropertyDescriptor(defaults, key);
        if (value && value.configurable && obj[key] === undefined) {
         Object.defineProperty(obj, key, value); 
         } 
       }
      return obj;
    }

    至此,IE9下总是报错的问题就解决了,希望能给同样掉进此坑的小伙伴一点帮助,早点摆脱IE的魔爪。

    参考文章:

    ES6 + Webpack + React + Babel 如何在低版本浏览器上愉快的玩耍(上)
    BABEL6 编译 ES6 继承代码的一个兼容问题(IE <= 10)

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-23 01:03 , Processed in 0.064986 second(s), 30 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表