成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

babel 到底將代碼轉(zhuǎn)換成什么鳥樣?

qc1iu / 1710人閱讀

摘要:大概就是將對象里面的一些屬性轉(zhuǎn)換成數(shù)組,方便解構(gòu)賦值的進行。而則更貼近的寫法,性能更好一些,兼容性更好一些,但將這部份代碼再轉(zhuǎn)換成的話會比較麻煩一些感覺這一點并不是缺點,有源碼就可以了。上面解決的辦法,實質(zhì)就是將改成。

原文鏈接:https://github.com/lcxfs1991/blog/issues/9

前言

將babel捧作前端一個劃時代的工具一定也不為過,它的出現(xiàn)讓許多程序員幸福地用上了es6新語法。但你就這么放心地讓babel跑在外網(wǎng)?反正我是不放心,我就曾經(jīng)過被坑過,于是萌生了研究babel代碼轉(zhuǎn)換的想法。本文不是分析babel源碼,僅僅是看看babel轉(zhuǎn)換的最終產(chǎn)物。

es6在babel中又稱為es2015。由于es2015語法眾多,本文僅挑選了較為常用的一些語法點,而且主要是分析babel-preset-2015這個插件(react開發(fā)的時候,常在webpack中用到這個preset)。

babel-preset-2015

打開babel-preset2015插件一看,一共20個插件。熟悉es2015語法的同志一看,多多少少能從字面意思知道某個插件是用于哪種語法的轉(zhuǎn)換

babel-plugin-transform-es2015-template-literals => es2015模板

babel-plugin-transform-es2015-literals

babel-plugin-transform-es2015-function-name => 函數(shù)name屬性

babel-plugin-transform-es2015-arrow-functions => 箭頭函數(shù)

babel-plugin-transform-es2015-block-scoped-functions => 函數(shù)塊級作用域

babel-plugin-transform-es2015-classes => class類

babel-plugin-transform-es2015-object-super => super提供了調(diào)用prototype的方式

babel-plugin-transform-es2015-shorthand-properties => 對象屬性的快捷定義,如obj = { x, y }

babel-plugin-transform-es2015-computed-properties => 對象中括號屬性,如obj = {["x]: 1}

babel-plugin-transform-es2015-for-of => 對象for of遍歷

babel-plugin-transform-es2015-sticky-regex

babel-plugin-transform-es2015-unicode-regex

babel-plugin-check-es2015-constants => const常量

babel-plugin-transform-es2015-spread => 對象擴展運算符屬性,如...foobar

babel-plugin-transform-es2015-parameters => 函數(shù)參數(shù)默認值及擴展運算符

babel-plugin-transform-es2015-destructuring => 賦值解構(gòu)

babel-plugin-transform-es2015-block-scoping => let和const塊級作用域

babel-plugin-transform-es2015-typeof-symbol => symbol特性

babel-plugin-transform-es2015-modules-commonjs => commonjs模塊加載

babel-plugin-transform-regenerator => generator特性

var, const and let

const和let現(xiàn)在一律轉(zhuǎn)換成var。那const到底如何保證不變呢?如果你在源碼中第二次修改const常量的值,babel編譯會直接報錯。
轉(zhuǎn)換前

var a = 1;
let b = 2;
const c = 3;

轉(zhuǎn)換后:

var a = 1;
var b = 2;
var c = 3;

那let的塊級作用怎么體現(xiàn)呢?來看看下面例子,實質(zhì)就是在塊級作用改變一下變量名,使之與外層不同。
轉(zhuǎn)換前:

let a1 = 1;
let a2 = 6;

{
    let a1 = 2;
    let a2 = 5;

    {
        let a1 = 4;
        let a2 = 5;
    }
}
a1 = 3;

轉(zhuǎn)換后:

var a1 = 1;
var a2 = 6;

{
    var _a = 2;
    var _a2 = 5;

    {
        var _a3 = 4;
        var _a4 = 5;
    }
}
a1 = 3;
賦值解構(gòu)

寫react的時候,我們使用負值解構(gòu)去取對象的值,用起來非常爽,像這樣:

var props = {
    name: "heyli",
    getName: function() {

    },
    setName: function() {

    }
};

let { name, getName, setName } = this.props;

我們來看看轉(zhuǎn)換的結(jié)果:

var props = {
    name: "heyli",
    getName: function getName() {},
    setName: function setName() {}
};

var name = props.name;
var getName = props.getName;
var setName = props.setName;

至于數(shù)組呢?如果是一個匿名數(shù)組,則babel會幫你先定義一個變量存放這個數(shù)組,然后再對需要賦值的變量進行賦值。
轉(zhuǎn)換前:

var [ a1, a2 ] = [1, 2, 3];

轉(zhuǎn)換后:

var _ref = [1, 2, 3];
var a1 = _ref[0];
var a2 = _ref[1];

看到這個,感覺轉(zhuǎn)換結(jié)果跟我們想的還蠻一致。哈哈,使用的噩夢還沒開始。

如果使用匿名對象直接進行賦值解構(gòu)會怎樣呢?如下。babel為了使接收的變量唯一,直接就將匿名對象里的屬性拼在一起,組成接收這個匿名對象的變量,嚇得我趕緊檢查一下項目里有沒有這種寫法。
轉(zhuǎn)換前:

var { abc, bcd, cde, def } = { "abc": "abc", "bcd": "bcd", "cde": "cde", "def": "def", "efg": "efg", "fgh": "fgh" };

轉(zhuǎn)換后:

var _abc$bcd$cde$def$efg$ = { "abc": "abc", "bcd": "bcd", "cde": "cde", "def": "def", "efg": "efg", "fgh": "fgh" };
var abc = _abc$bcd$cde$def$efg$.abc;
var bcd = _abc$bcd$cde$def$efg$.bcd;
var cde = _abc$bcd$cde$def$efg$.cde;
var def = _abc$bcd$cde$def$efg$.def;

還有一種對象深層次的解構(gòu)賦值:
轉(zhuǎn)換前:

var obj = {
    p1: [
        "Hello",
        { p2: "World" }
      ]
};

var { p1: [s1, { p2 }] } = obj;

轉(zhuǎn)換后:

// 為解釋本人將代碼美化了
var _slicedToArray = (function() {
    function sliceIterator(arr, i) {
        var _arr = [];
        var _n = true;
        var _d = false;
        var _e = undefined;
        try {
           // 用Symbol.iterator造了一個可遍歷對象,然后進去遍歷。
            for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) {
                _arr.push(_s.value);
                if (i && _arr.length === i) break;
            }
        } catch (err) {
            _d = true;
            _e = err;
        } finally {
            try {
                if (!_n && _i["return"]) _i["return"]();
            } finally {
                if (_d) throw _e;
            }
        }
        return _arr;
    }
    return function(arr, i) {
        if (Array.isArray(arr)) {
            return arr;
        } else if (Symbol.iterator in Object(arr)) {
            return sliceIterator(arr, i);
        } else {
            throw new TypeError("Invalid attempt to destructure non-iterable instance");
        }
    };
})();

var obj = {
   p1: ["Hello", { p2: "World" }]
};

var _obj$p = _slicedToArray(obj.p1, 2);

var s1 = _obj$p[0];
var p2 = _obj$p[1].p2;

babel在代碼頂部生產(chǎn)了一個公共的代碼_slicedToArray。大概就是將對象里面的一些屬性轉(zhuǎn)換成數(shù)組,方便解構(gòu)賦值的進行。但Symbol.iterator的兼容性并不好(如下圖),還是謹慎使用為妙。

另外,下面這種對字符串進行賦值解構(gòu)也同樣使用到_slicedToArray方法:

const [a, b, c, d, e] = "hello";
函數(shù)參數(shù)默認值及擴展運算符

在es5的年代,一般我們寫參數(shù)的默認值都會這么寫:

function func(x, y) {
    var x = x || 1;
    var y = y || 2;
}

我們來看看babel的轉(zhuǎn)換辦法:

function func({x, y} = { x: 0, y: 0 }) {
  return [x, y];
}

function func1(x = 1, y = 2) {
    return [x, y];
}
function func() {
  var _ref = arguments.length <= 0 || arguments[0] === undefined ? { x: 0, y: 0 } : arguments[0];

  var x = _ref.x;
  var y = _ref.y;

  return [x, y];
}

function func1() {
  var x = arguments.length <= 0 || arguments[0] === undefined ? 1 : arguments[0];
  var y = arguments.length <= 1 || arguments[1] === undefined ? 2 : arguments[1];

  return [x, y];
}

babel這里使有了arguments來做判。第一種情況涉及解構(gòu)賦值,因此x和y的值還是有可能是undefined的。至于第二種情況,則會保證2個參數(shù)的默認值分別是1和2.

再來看一種。...y代表它接收了剩下的參數(shù)。也就是arguments除了第一個標號的參數(shù)之外剩余的參數(shù)。

轉(zhuǎn)換前:

function func(x, ...y) {
    console.log(x);
    console.log(y);
    return x * y.length;
}

轉(zhuǎn)換后:

function func(x) {
    console.log(x);

    for (var _len = arguments.length, y = Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {
        y[_key - 1] = arguments[_key];
    }

    console.log(y);
    return x * y.length;
}
箭頭函數(shù)

剪頭函數(shù)其實主要是省了寫函數(shù)的代碼,同時能夠直接用使外層的this而不用擔心context切換的問題。以前我們一般都要在外層多寫一個_this/self直向this。babel的轉(zhuǎn)換辦法其實跟我們的處理無異。

轉(zhuǎn)換前:

var obj = {
    prop: 1,
    func: function() {
        var _this = this;

        var innerFunc = () => {
            this.prop = 1;
        };

        var innerFunc1 = function() {
            this.prop = 1;
        };
    },

};

轉(zhuǎn)換后:

var obj = {
    prop: 1,
    func: function func() {
        var _this2 = this;

        var _this = this;

        var innerFunc = function innerFunc() {
            _this2.prop = 1;
        };

        var innerFunc1 = function innerFunc1() {
            this.prop = 1;
        };
    }

};
對象的能力增強 對象屬性的快捷定義

轉(zhuǎn)換前:

var a = 1,
    b = "2",
    c = function() {
        console.log("c");
    };

var obj = {a, b, c};

轉(zhuǎn)換后:

var a = 1,
    b = "2",
    c = function c() {
    console.log("c");
};

var obj = { a: a, b: b, c: c };
對象中括號屬性

es2015開始新增了在對象中用中括號解釋屬性的功能,這對變量、常量等當對象屬性尤其有用。

轉(zhuǎn)換前:

const prop2 = "PROP2";
var obj = {
    ["prop"]: 1,
    ["func"]: function() {
        console.log("func");
    },
        [prop2]: 3
};

轉(zhuǎn)換后:

var _obj;
// 已美化
function _defineProperty(obj, key, value) {
    if (key in obj) {
        Object.defineProperty(obj, key, {
            value: value,
            enumerable: true,
            configurable: true,
            writable: true
        });
    } else {
        obj[key] = value;
    }
    return obj;
}

var prop2 = "PROP2";
var obj = (_obj = {}, _defineProperty(_obj, "prop", 1), _defineProperty(_obj, "func", function func() {
    console.log("func");
}), _defineProperty(_obj, prop2, 3), _obj);

看似簡單的屬性,babel卻大動干戈。新增了一個_defineProperty函數(shù),給新建的_obj = {}進行屬性定義。除此之外使用小括號包住一系列從左到右的運算使整個定義更簡潔。

使用super去調(diào)用prototype

以前我們一般都用obj.prototype或者嘗試用this去往上尋找prototype上面的方法。而babel則自己寫了一套在prototype鏈上尋找方法/屬性的算法。

轉(zhuǎn)換前:

var obj = {
    toString() {
     // Super calls
     return "d " + super.toString();
    },
};

轉(zhuǎn)換后:

var _obj;
// 已美化
var _get = function get(object, property, receiver) {
   // 如果prototype為空,則往Function的prototype上尋找
    if (object === null) object = Function.prototype;
    var desc = Object.getOwnPropertyDescriptor(object, property);
    if (desc === undefined) {
        var parent = Object.getPrototypeOf(object);
        // 如果在本層prototype找不到,再往更深層的prototype上找
        if (parent === null) {
            return undefined;
        } else {
            return get(parent, property, receiver);
        }
    }
    // 如果是屬性,則直接返回
    else if ("value" in desc) {
        return desc.value;
    }
    // 如果是方法,則用call來調(diào)用,receiver是調(diào)用的對象 
    else {
        var getter = desc.get;  // getOwnPropertyDescriptor返回的getter方法
        if (getter === undefined) {
            return undefined;
        }
        return getter.call(receiver);
    }
};

var obj = _obj = {
  toString: function toString() {
    // Super calls
    return "d " + _get(Object.getPrototypeOf(_obj), "toString", this).call(this);
  }
};
Object.assign 和 Object.is

es6新增的Object.assign極大方便了對象的克隆復(fù)制。但babel的es2015 preset并不支持,所以沒對其進入轉(zhuǎn)換,這會使得一些移動端機子遇到這種寫法會報錯。所以一般開發(fā)者都會使用object-assign這個npm的庫做兼容。

Object.is用于比較對象的值與類型,es2015 preset同樣不支持編譯。

es6模板 多行字符串

轉(zhuǎn)換前:

console.log(`string text line 1
string text line 2`);

轉(zhuǎn)換后:

console.log("string text line 1
string text line 2");
字符中變量運算

轉(zhuǎn)換前:

var a = 5;
var b = 10;
console.log(`Fifteen is ${a + b} and not ${2 * a + b}.`);

轉(zhuǎn)換后:

var a = 5;
var b = 10;
console.log("Fifteen is " + (a + b) + " and not " + (2 * a + b) + ".");
標簽?zāi)0?/b>

es6的這種新特性給模板處理賦予更強大的功能,一改以往對模板進行各種replace的處理辦法,用一個統(tǒng)一的handler去處理。babel的轉(zhuǎn)換主要是添加了2個屬性,因此看起來也并不算比較工程浩大的編譯。

轉(zhuǎn)換前:

var a = 5;
var b = 10;

function tag(strings, ...values) {
  console.log(strings[0]); // "Hello "
  console.log(strings[1]); // " world "
  console.log(values[0]);  // 15
  console.log(values[1]);  // 50

  return "Bazinga!";
}

tag`Hello ${ a + b } world ${ a * b }`;

轉(zhuǎn)換后:

var _templateObject = _taggedTemplateLiteral(["Hello ", " world ", ""], ["Hello ", " world ", ""]);
// 已美化
function _taggedTemplateLiteral(strings, raw) {
    return Object.freeze(Object.defineProperties(strings, {
        raw: {
            value: Object.freeze(raw)
        }
    }));
}
// 給傳入的object定義strings和raw兩個不可變的屬性。

var a = 5;
var b = 10;

function tag(strings) {
  console.log(strings[0]); // "Hello "
  console.log(strings[1]); // " world "
  console.log(arguments.length <= 1 ? undefined : arguments[1]); // 15
  console.log(arguments.length <= 2 ? undefined : arguments[2]); // 50

  return "Bazinga!";
}

tag(_templateObject, a + b, a * b);
模塊化與類 類class

javascript實現(xiàn)oo一直是非常熱門的話題。從最原始時代需要手動維護在構(gòu)造函數(shù)里調(diào)用父類構(gòu)造函數(shù),到后來封裝好函數(shù)進行extend繼承,再到babel出現(xiàn)之后可以像其它面向?qū)ο蟮恼Z言一樣直接寫class。es2015的類方案仍然算是過渡方案,它所支持的特性仍然沒有涵蓋類的所有特性。目前主要支持的有:

constructor

static方法

get 方法

set 方法

類繼承

super調(diào)用父類方法。

轉(zhuǎn)換前:

class Animal {

    constructor(name, type) {
        this.name = name;
        this.type = type;
    }

    walk() {
        console.log("walk");
    }

    run() {
        console.log("run")
    }

    static getType() {
        return this.type;
    }

    get getName() {
        return this.name;
    }

    set setName(name) {
        this.name = name;
    }


}

class Dog extends Animal {
    constructor(name, type) {
        super(name, type);
    }

    get getName() {
        return super.getName();
    }
}

轉(zhuǎn)換后(由于代碼太長,先省略輔助的方法):

/**
......一堆輔助方法,后文詳述
**/
var Animal = (function () {
    function Animal(name, type) {
                // 此處是constructor的實現(xiàn),用_classCallCheck來判定constructor正確與否
        _classCallCheck(this, Animal);

        this.name = name;
        this.type = type;
    }
        // _creatClass用于創(chuàng)建類及其對應(yīng)的方法
    _createClass(Animal, [{
        key: "walk",
        value: function walk() {
            console.log("walk");
        }
    }, {
        key: "run",
        value: function run() {
            console.log("run");
        }
    }, {
        key: "getName",
        get: function get() {
            return this.name;
        }
    }, {
        key: "setName",
        set: function set(name) {
            this.name = name;
        }
    }], [{
        key: "getType",
        value: function getType() {
            return this.type;
        }
    }]);

    return Animal;
})();

var Dog = (function (_Animal) {
        // 子類繼承父類
    _inherits(Dog, _Animal);

    function Dog(name, type) {
        _classCallCheck(this, Dog);
                // 子類實現(xiàn)constructor
                // babel會強制子類在constructor中使用super,否則編譯會報錯
        return _possibleConstructorReturn(this, Object.getPrototypeOf(Dog).call(this, name, type));
    }

    _createClass(Dog, [{
        key: "getName",
        get: function get() {
                       // 跟上文使用super調(diào)用原型鏈的super編譯解析的方法一致,
                       // 也是自己寫了一個回溯prototype原型鏈
            return _get(Object.getPrototypeOf(Dog.prototype), "getName", this).call(this);
        }
    }]);

    return Dog;
})(Animal);
// 檢測constructor正確與否
function _classCallCheck(instance, Constructor) {
    if (!(instance instanceof Constructor)) {
        throw new TypeError("Cannot call a class as a function");
    }
}
// 創(chuàng)建類
var _createClass = (function() {
    function defineProperties(target, props) {
        for (var i = 0; i < props.length; i++) {
            var descriptor = props[i];
            // es6規(guī)范要求類方法為non-enumerable
            descriptor.enumerable = descriptor.enumerable || false;
            descriptor.configurable = true;
            // 對于setter和getter方法,writable為false
            if ("value" in descriptor) descriptor.writable = true;
            Object.defineProperty(target, descriptor.key, descriptor);
        }
    }
    return function(Constructor, protoProps, staticProps) {
        // 非靜態(tài)方法定義在原型鏈上
        if (protoProps) defineProperties(Constructor.prototype, protoProps);
        // 靜態(tài)方法直接定義在constructor函數(shù)上
        if (staticProps) defineProperties(Constructor, staticProps);
        return Constructor;
    };
})();
// 繼承類
function _inherits(subClass, superClass) {
   // 父類一定要是function類型
    if (typeof superClass !== "function" && superClass !== null) {
        throw new TypeError("Super expression must either be null or a function, not " + typeof superClass);
    }
    // 使原型鏈subClass.prototype.__proto__指向父類superClass,同時保證constructor是subClass自己
    subClass.prototype = Object.create(superClass && superClass.prototype, {
        constructor: {
            value: subClass,
            enumerable: false,
            writable: true,
            configurable: true
        }
    });
    // 保證subClass.__proto__指向父類superClass
    if (superClass) 
        Object.setPrototypeOf ? 
        Object.setPrototypeOf(subClass, superClass) :    subClass.__proto__ = superClass;
}
// 子類實現(xiàn)constructor
function _possibleConstructorReturn(self, call) {
    if (!self) {
        throw new ReferenceError("this hasn"t been initialised - super() hasn"t been called");
    }
    // 若call是函數(shù)/對象則返回
    return call && (typeof call === "object" || typeof call === "function") ? call : self;
}

先前在用react重構(gòu)項目的時候,所有的react組件都已經(jīng)摒棄了es5的寫法,一律采用了es6。用類的好處寫繼續(xù)更加方便,但無法用mixin,需要借助更新的es7語法中的decorator才能夠?qū)崿F(xiàn)類mixin的功能(例如pureRender)。但這次分析完babel源碼之后,才發(fā)現(xiàn)原來babel在實現(xiàn)class特性的時候,定義了許多方法,盡管看起來并不太優(yōu)雅。

模塊化

在開發(fā)react的時候,我們往往用webpack搭配babel的es2015和react兩個preset進行構(gòu)建。之前看了一篇文章對babel此處的模塊加載有些啟發(fā)(《分析 Babel 轉(zhuǎn)換 ES6 module 的原理》)。

示例:

// test.js
import { Animal as Ani, catwalk } from "./t1";
import * as All from "./t2";


class Cat extends Ani {

    constructor() {
        super();
    }
}

class Dog extends Ani {
    constructor() {
        super();
    }
}
// t1.js
export class Animal {

    constructor() {

    }

}

export function catwal() {
    console.log("cat walk");
};
// t2.js
export class Person {
    constructor() {

    }

}

export class Plane {
    constructor() {

    }

}

通過webpack與babel編譯后:

// t1.js的模塊
Object.defineProperty(exports, "__esModule", {
    value: true
});
exports.catwal = catwal;

// 省略一些類繼承的方法

var Animal = exports.Animal = function Animal() {
    _classCallCheck(this, Animal);
};

function catwal() {
    console.log("cat walk");
};

// t2.js的模塊
Object.defineProperty(exports, "__esModule", {
    value: true
});

// 省略一些類繼承的方法

var Person = exports.Person = function Person() {
    _classCallCheck(this, Person);
};

var Plane = exports.Plane = function Plane() {
    _classCallCheck(this, Plane);
};

// test.js的模塊
var _t = __webpack_require__(1);

var _t2 = __webpack_require__(3); // 返回的都是exports上返回的對象屬性

var All = _interopRequireWildcard(_t2);

function _interopRequireWildcard(obj) {
    // 發(fā)現(xiàn)是babel編譯的, 直接返回
    if (obj && obj.__esModule) {
        return obj;
    }
   // 非babel編譯, 猜測可能是第三方模塊,為了不報錯,讓default指向它自己
    else {
        var newObj = {};
        if (obj != null) {
            for (var key in obj) {
                if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key];
            }
        }
        newObj.default = obj;
        return newObj;
    }
}

// 省略一些類繼承的方法

var Cat = (function (_Ani) {
    _inherits(Cat, _Ani);

    function Cat() {
        _classCallCheck(this, Cat);

        return _possibleConstructorReturn(this, Object.getPrototypeOf(Cat).call(this));
    }

    return Cat;
})(_t.Animal);

var Dog = (function (_Ani2) {
    _inherits(Dog, _Ani2);

    function Dog() {
        _classCallCheck(this, Dog);

        return _possibleConstructorReturn(this, Object.getPrototypeOf(Dog).call(this));
    }

    return Dog;
})(_t.Animal);

es6的模塊加載是屬于多對象多加載,而commonjs則屬于單對象單加載。babel需要做一些手腳才能將es6的模塊寫法寫成commonjs的寫法。主要是通過定義__esModule這個屬性來判斷這個模塊是否經(jīng)過babel的編譯。然后通過_interopRequireWildcard對各個模塊的引用進行相應(yīng)的處理。

另一個發(fā)現(xiàn)是,通過webpack打包babel編譯后的代碼,每一個模塊里面都包含了相同的類繼承幫助方法,這是開發(fā)時忽略的。由此可看,在開發(fā)react的時候用es5的語法可能會比使用es6的class能使js bundle更小。

babel es2015 loose mode

開發(fā)家校群的時候,在android4.0下面報esModule錯誤的問題,如下:
Uncaught TypeError: Cannot assign to read only property "__esModule" of #。

經(jīng)查證,發(fā)現(xiàn)是構(gòu)建中babel-es2015 loader的模式問題,會導(dǎo)致Android4.0的用戶有報錯。只需要使用loose mode就可以解決問題。下面是相關(guān)的stackoverflow issue以及對應(yīng)解決問題的npm包。

stackoverflow

babel-preset-est2015-loose npm package

那么es2015和normal mode和loose mode有什么區(qū)別呢,這個出名的博客略有介紹:Babel 6: loose mode。

實質(zhì)就是(作者總結(jié))normal mode的轉(zhuǎn)換更貼近es6的寫法,許多的property都是通過Object.defineProperty進行的。而loose mode則更貼近es5的寫法,性能更好一些,兼容性更好一些,但將這部份代碼再轉(zhuǎn)換成native es6的話會比較麻煩一些(感覺這一點并不是缺點,有源碼就可以了)。

上面esModule解決的辦法,實質(zhì)就是將

Object.defineProperty(exports, "__esModule", {
    value: true
});

改成 exports.__esModule = true;

再舉個例子,如下面的Cat類定義:

class Cat extends Ani {

    constructor() {
        super();
    }

    miao() {
        console.log("miao");
    }
}

正常模式會編譯為:

var Cat = (function (_Ani) {
    _inherits(Cat, _Ani);

    function Cat() {
        _classCallCheck(this, Cat);

        return _possibleConstructorReturn(this, Object.getPrototypeOf(Cat).call(this));
    }

    _createClass(Cat, [{
        key: "miao",
        value: function miao() {
            console.log("miao");
        }
    }]);

    return Cat;
})(_t.Animal);

loose mode模式會編譯為:

var Cat = (function (_Ani) {
    _inherits(Cat, _Ani);

    function Cat() {
        _classCallCheck(this, Cat);

        return _possibleConstructorReturn(this, _Ani.call(this));
    }

    Cat.prototype.miao = function miao() {
        console.log("miao");
    };

    return Cat;
})(_t.Animal);

babel es2015中l(wèi)oose模式主要是針對下面幾個plugin:

transform-es2015-template-literals

transform-es2015-classes

transform-es2015-computed-properties

transform-es2015-for-of

transform-es2015-spread

transform-es2015-destructuring

transform-es2015-modules-commonjs

每一種的轉(zhuǎn)換方式在此就不再贅述了,大家可以回家自己試。

如有錯誤,懇請斧正!

參考文章 babel try out

https://babeljs.io/repl/

template literals

https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Template_literals

block-scoped functions

https://blogs.msdn.microsoft.com/cdndevs/2015/09/16/future-of-javascript-ecmascript-6-es2015-block-scoping/

http://www.programmerinterview.com/index.php/javascript/javascript-block-scope/

http://www.2ality.com/2015/02/es6-scoping.html

classes

http://purplebamboo.github.io/2014/07/13/javascript-oo-class/

http://blog.rainy.im/2015/07/20/prototype-chain-in-js/

objects

http://fourkitchens.com/blog/article/practical-introduction-es2015-objects

commonjs and es6 module

http://ryerh.com/javascript/2016/03/27/babel-module-implementation.html

http://www.2ality.com/2015/12/babel6-loose-mode.html

http://www.2ality.com/2015/02/es6-classes-final.html

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/79432.html

相關(guān)文章

  • ES6部分方法點評(三):babel-preset-es2015-loose可轉(zhuǎn)換且移動端兼容性較好

    摘要:是目前最常用的轉(zhuǎn)的工具,但即使是,各瀏覽器的支持度也是不一的,因此便產(chǎn)生了本文,找出一些能夠被翻譯成兼容性高代碼的語法。不可用因為會使用到參考資料,用來查瀏覽器兼容性。到底將代碼轉(zhuǎn)換成什么鳥樣,感謝。官網(wǎng)試驗轉(zhuǎn)換后的代碼 前言 由于目前各瀏覽器對ES6兼容性較低,再加上需要兼容歷史上各種版本的瀏覽器,因此,使用編譯器將ES6語法轉(zhuǎn)譯成ES5語法則勢在必行了。babel是目前最常用的ES...

    wean 評論0 收藏0
  • 我他喵的到底要怎樣才能在生產(chǎn)環(huán)境中用上 ES6 模塊化?

    摘要:因此,你還是需要各種各樣雜七雜八的工具來轉(zhuǎn)換你的代碼噢,我可去你媽的吧,這些東西都是干嘛的我就是想用個模塊化,我到底該用啥子本文正旨在列出幾種可用的在生產(chǎn)環(huán)境中放心使用模塊化的方法,希望能幫到諸位后來者這方面的中文資源實在是忒少了。 原文發(fā)表在我的博客上。最近搗鼓了一下 ES6 的模塊化,分享一些經(jīng)驗 :) Python3 已經(jīng)發(fā)布了九年了,Python 社區(qū)卻還在用 Python 2...

    KaltZK 評論0 收藏0
  • 【用故事解讀 MobX 源碼(四)】裝飾器 和 Enhancer

    摘要:所以這是一篇插隊的文章,用于去理解中的裝飾器和概念。因此,該的作用就是根據(jù)入?yún)⒎祷鼐唧w的描述符。其次局部來看,裝飾器具體應(yīng)用表達式是,其函數(shù)簽名和是一模一樣。等裝飾器語法,是和直接使用是等效等價的。 ================前言=================== 初衷:以系列故事的方式展現(xiàn) MobX 源碼邏輯,盡可能以易懂的方式講解源碼; 本系列文章: 《【用故事解...

    maybe_009 評論0 收藏0
  • babel學習筆記

    摘要:經(jīng)過一番折騰,總算是把自己項目里的配置調(diào)整好了,所有文件從原來的縮小到。折騰了不少時間,改動其實就一個地方,就是配置文件,記錄一下自己折騰的過程。本以為那這兩種方式取其一就行了。這感覺和想象中的不一樣啊,說好的一個搞定一切的呢。。。 先是看到前端早讀課【第1065期】再見,babel-preset-2015,聽說現(xiàn)在有了babel-preset-env,別的什么preset都不需要了,...

    Aomine 評論0 收藏0
  • 在使用es6語法class的時候,babel到底做了什么?

    摘要:正常情況,的返回值就是一個對象,其實也就是對象。好了,上面算是基本說清楚了使用語法定義類繼承類,到底發(fā)生了什么,如果錯誤,還請指正,謝謝 自從有了webpack之后,我們這些jscoder似乎得到了前所未有的解放,箭頭函數(shù),對象解構(gòu),let,const關(guān)鍵字,以及class、extends等等關(guān)鍵字使用得不亦樂乎,反正,webpack會幫我們把這些es6代碼轉(zhuǎn)換成瀏覽器能夠識別的es5...

    時飛 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<