如何區分Babel中的stage-0,stage-1,stage-2以及stage-3(一)

你們知道,將ES6代碼編譯爲ES5時,咱們經常使用到Babel這個編譯工具。你們參考一些網上的文章或者官方文檔,裏面常會建議你們在.babelrc中輸入以下代碼:javascript

複製代碼
{
    "presets": [
      "es2015",
      "react",
      "stage-0"
    ],
    "plugins": []
  }
複製代碼

 

咱們如今來講明下這個配置文件是什麼意思。首先,這個配置文件是針對babel 6的。Babel 6作了一系列模塊化,不像Babel 5同樣把全部的內容都加載。好比須要編譯ES6,咱們須要設置presets爲"es2015",也就是預先加載es6編譯的相關模塊,若是須要編譯jsx,須要預先加載"react"這個模塊。那問題來了,這個"stage-0"又表明什麼呢? 有了"react-0",是否又有諸如"stage-1", "stage-2"等等呢?java

 

事實上, 」stage-0"是對ES7一些提案的支持,Babel經過插件的方式引入,讓Babel能夠編譯ES7代碼。固然因爲ES7沒有定下來,因此這些功能隨時肯能被廢棄掉的。如今咱們來一一分析裏面都有什麼。node

 

1. 法力無邊的stage-0react

 

爲何說「stage-0」 法力無邊呢,由於它包含stage-1, stage-2以及stage-3的全部功能,同時還另外支持以下兩個功能插件:webpack

 

用過React的同窗可能知道,jsx對條件表達式支持的不是太好,你不能很方便的使用if/else表達式,要麼你使用三元表達,要麼用函數。例如你不能寫以下的代碼:git

 

複製代碼
var App = React.createClass({

    render(){
        let { color } = this.props;

        return (
            <div className="parents">
                {
                    if(color == 'blue') { 
                        <BlueComponent/>; 
                    }else if(color == 'red') { 
                        <RedComponent/>; 
                    }else { 
                        <GreenComponent/>; }
                    }
                }
            </div>
        )
    }
})
複製代碼

 

在React中你只能寫成這樣:es6

複製代碼
var App = React.createClass({

    render(){
        let { color } = this.props;


        const getColoredComponent = color => {
            if(color === 'blue') { return <BlueComponent/>; }
            if(color === 'red') { return <RedComponent/>; }
            if(color === 'green') { return <GreenComponent/>; }
        }


        return (
            <div className="parents">
                { getColoredComponent(color) }
            </div>
        )
    }
})
複製代碼

 

transform-do-expressions 這個插件就是爲了方便在 jsx寫if/else表達式而提出的,咱們能夠重寫下代碼。github

複製代碼
var App = React.createClass({

    render(){
        let { color } = this.props;

        return (
            <div className="parents">
                {do {
                    if(color == 'blue') { 
                        <BlueComponent/>; 
                    }else if(color == 'red') { 
                        <RedComponent/>; 
                    }else { 
                        <GreenComponent/>; }
                    }
                }}
            </div>
        )
    }
})
複製代碼

 

再說說 transform-function-bind, 這個插件其實就是提供過 :: 這個操做符來方便快速切換上下文, 以下面的代碼:web

複製代碼
obj::func
// is equivalent to:
func.bind(obj)

obj::func(val)
// is equivalent to:
func.call(obj, val)

::obj.func(val)
// is equivalent to:
func.call(obj, val)

// 再來一個複雜點的樣例

const box = {
  weight: 2,
  getWeight() { return this.weight; },
};

const { getWeight } = box;

console.log(box.getWeight()); // prints '2'

const bigBox = { weight: 10 };
console.log(bigBox::getWeight()); // prints '10'

// Can be chained:
function add(val) { return this + val; }

console.log(bigBox::getWeight()::add(5)); // prints '15'
複製代碼

 

若是想更屌點,還能夠寫出更牛逼的代碼:express

複製代碼
const { map, filter } = Array.prototype;

let sslUrls = document.querySelectorAll('a')
                ::map(node => node.href)
                ::filter(href => href.substring(0, 5) === 'https');

console.log(sslUrls);
複製代碼

 

2. 一應俱全的stage-1

stage-1除了包含stage-2和stage-3,還包含了下面4個插件:

 

今天就到這裏了,改天咱們接着分析下"stage-2"和"stage-3", 它們又有什麼新的特性呢,讓咱們拭目以待吧。

 

 

 

 

 

 

 

深藏不露的stage-2

爲何說 stage-2深藏不露呢,由於它很低調,低調到你能夠忽略它,但事實上,它頗有內涵的。它除了覆蓋stage-3的全部功能,還支持以下兩個插件:

syntax-trailing-function-commas

這個插件讓人一看以爲挺沒趣的,讓人甚至以爲它有點雞肋。因它不是對ES6功能的增長,而是爲了加強代碼的可讀性和可修改性而提出的。以下面的代碼所示:

// 假設有以下的一個函數,它有兩個參數 function clownPuppiesEverywhere( param1, param2 ) { /* ... */ } clownPuppiesEverywhere( 'foo', 'bar' ); // 有一天,它須要變成3個參數,你須要這樣修改 function clownPuppiesEverywhere( param1, - param2 + param2, // 這一行得加一個逗號 + param3 // 增長參數param3 ) { /* ... */ } clownPuppiesEverywhere( 'foo', - 'bar' + 'bar', // 這裏的修改成逗號 + 'baz' // 增長新的參數 ); // 看到沒? 咱們修改了4行代碼。。啊啊。修改了4行代碼。

修改了4行代碼,嗯嗯嗯。。追求高效的程序猿想一想了,之後若是有更多參數了,我是否是要改等多行,得想一想,代碼改的越少越好,因而有了下面的改動。。

// 咱們來從新定義一下函數 function clownPuppiesEverywhere( param1, param2, // 注意這裏,咱們加了一個逗號喲 ) { /* ... */ } clownPuppiesEverywhere( 'foo', 'bar', // 這裏咱們也加了一個逗號 ); // 如今函數須要三個參數,咱們來修改下 function clownPuppiesEverywhere( param1, param2, + param3, // 增長params3參數 ) { /* ... */ } clownPuppiesEverywhere( 'foo', 'bar', + 'baz', // 增長第三個參數 ); // 叮叮噹,咱們只修改了兩行代碼就完成了,好開森

說實話吧,這個功能讓人有點很無語。不過程序猿對乾淨代碼的追求真的很讓人感動,仍是值得鼓勵的。這個就是stage-2中"尾逗號函數」功能。哈哈哈哈。

transform-object-rest-spread

再來講transform-object-rest-spread, 其實它是對 ES6中解構賦值的一個擴展,由於ES6只支持對數組的解構賦值,對對象是不支持的。以下面的代碼所示:

// 獲取剩下的屬性 let { x, y, ...z } = { x: 1, y: 2, a: 3, b: 4 }; console.log(x); // 1 console.log(y); // 2 console.log(z); // { a: 3, b: 4 } // 屬性展開 let n = { x, y, ...z }; console.log(n); // { x: 1, y: 2, a: 3, b: 4 }

大放異彩的stage3

爲啥說stage3大放異彩呢?由於它支持大名鼎鼎的asyncawait, 這兩個哥們但是解決(Ajax)回調函數的終極解決方法呀!管你什麼異步,我均可以用同步的思惟來寫,ES7裏面很是強悍的存在。總的來講,它包含以下兩個插件:

transform-async-to-generator

transform-async-to-generator主要用來支持ES7中的asyncawait, 咱們能夠寫出下面的代碼:

const sleep = (timeout)=>{ return new Promise( (resolve, reject)=>{ setTimeout(resolve, timeout) }) } (async ()=>{ console.time("async"); await sleep(3000); console.timeEnd("async"); })()

再來一個實際點的例子

const fetchUsers = (user)=>{ return window.fetch(`https://api.douban.com/v2/user/${user}`).then( res=>res.json()) } const getUser = async (user) =>{ let users = await fetchUsers(user); console.log( users); } console.log( getUser("flyingzl"))

提示: 因爲asycnawait是ES7裏面的內容,現階段不建議使用。爲了順利運行上面的代碼,建議用webpack進行編譯。

transform-exponentiation-operator

transform-exponentiation-operator這個插件算是一個語法糖,能夠經過**這個符號來進行冪操做,想當於Math.pow(a,b)。以下面的樣例

// x ** y let squared = 2 ** 2; // 至關於: 2 * 2 let cubed = 2 ** 3; // 至關於: 2 * 2 * 2 // x **= y let a = 2; a **= 2; // 至關於: a = a * a; let b = 3; b **= 3; // 至關於: b = b * b * b;

很簡單也很實用吧,哈。使用起來仍是蠻方便的。

總結

經過以上的兩篇文章,咱們瞭解了stage-0,state-1stage-2以及stage-3的區別。在進行實際開發時,能夠更具須要來設置對應的stage。若是省事懶得折騰,通常設置爲stage-0便可。若是爲了防止開發人員使用某些太新的功能,咱們能夠限制到某個特定的stage便可。若是有其餘問題,歡迎你們留言 :-)。 更詳細的請參考https://babeljs.io/docs/plugins/preset-stage-0/

相關文章
相關標籤/搜索