React Conf 2017 干货总结1: React + ES next = ♥

React Conf 2017在加利福尼亚州的圣克拉拉万豪酒店圆满落幕,这已经是Facebook举办的第三届React官方大会了。
虽然不能参会,但是作为前端开发者,我们当然不能错过这个绝佳的学习契机。

笔者利用清明假期,参看了YouTube上关于这次大会的记录。一共34个精彩演讲,对应34个视频。获益匪浅。

这里,将会作为一个系列,对其中的几篇演讲进行翻译和分析。并辅助以code demo,帮助大家理解。

欢迎关注我的简书掘金账号,也欢迎在Github上follow,最新的大会code demo,便可第一时间掌握。

今天为大家介绍的是Ben Ilegbodu的主题:**React + ES next = ♥ **

Ben Ilegbodu是“为数不多的”参会有色程序员,黑人程序员如同女性程序员一样凤毛麟角。但是,本次分享主题很有爱,很有营养,精彩程度丝毫不打折扣。如果你不了解React也不要紧,因为这次讲的是ES6、ES7在React中的应用。所以,其实是对下一代ES的普及和介绍。

本文将以

  • Destructuring、
  • Spread Opetator、
  • Arrow Function、
  • Promises、
  • Async Functions

这几方面展开。并通过nodeJS实现一个兼具前端和后端的小型“评论/留言 发布阅读系统”。

建议看这篇文章的同时,结合视频一起研究:Ben Ilegbodu - React + ES next = ♥ - React Conf 2017

实现预览

如图,我们实现了如下的页面。这是一个前端+后端的全栈小项目。
当然,样式是极其简陋的。作为“粗糙”的程序员,实在懒得在页面上花时间。

es-next.png

我们可以在输入框内输入姓名和留言内容。并点击按钮提交。后台使用nodeJS express框架,实现对文件的读写更新。

app.post('/api/comments', function(req, res) {
    fs.readFile(COMMENTS_FILE, function(err, data) {
        if (err) {
            console.error(err);
            process.exit(1);
        }
        var comments = JSON.parse(data);
        var newComment = {
            id: Date.now(),
            author: req.body.author,
            text: req.body.text,
        };
        comments.push(newComment);
        fs.writeFile(COMMENTS_FILE, JSON.stringify(comments, null, 4), function(err) {
            if (err) {
                console.error(err);
                process.exit(1);
            }
            res.json(comments);
        });
    });
});

这是nodeJS的后端代码,如果不理解也没关系。

接下来我们继续回到演讲。

解构Destructuring

在我们的项目中,最初版存在这样的一段代码:

_handleCommentSubmit(comment) {
    let comments = this.state.comments;
    ...
    // remaining code
}

请务必记住这个_handleCommentSubmit函数,接下来的全文都是围绕他展开,并进行一步步拓展。
这个函数的逻辑是对新提交的comment进行处理,首先他需要从state中读取现有的comments数组。
在使用解构赋值的情况下,我们重构为:

_handleCommentSubmit(comment) {
    let {comments} = this.state;
    ...
    // remaining code
}

也许这还看不出来解构到底有什么作用,但是在逻辑多的时候,他是很有必要的。比如:

let author = this.state.author;
let text = this.state.text;

就可以写为:

let {author, text} = this.state;

如果需要改变变量名时,就可以从:

let authorName = this.state.author;
let fullText = this.state.text;

改为:

let {author: authorName, text: fullText} = this.state;

再举一个例子,在function component(React无状态组件编写的一种推荐形式)情况下:

function MyComponent(props) {
    return (
        <div style={props.style}>{props.children}</div>
    )
}
<MyComponent style="dark">Stateless function!</MyComponent>

我们可以改写为:

function MyComponent({children, style}) {
    return (
        <div style={style}>{children}</div>
    )
}
<MyComponent style="dark">Stateless function!</MyComponent>

展开符Spread Opetator

还记得上面那个_handleCommentSubmit函数吗?
接下来我们要进行扩充。首先我们将新提交的comment(即函数参数),添加一个时间戳作为id。接下来,
我们拿到旧的state.comments之后,就要将新的comment(包含id)加入到state.comments当中。

初版做法是:

_handleCommentSubmit(comment) {
    let {comments} = this.state;
    let newComment = comment;
    
    newComment.id = Date.now();

    let newComments = comments.concat([newComment]);

    ...
    // setState + ajax stuffs
}

在使用展开符后,我们可以重构为:

_handleCommentSubmit(comment) {
    let {comments} = this.state;
    let newComment = {...comment, id: Date.now()};
    let newComments = [...comments, newComment];
    ...
    // setState + ajax stuffs
}

当然,展开符还有很多其他benefits;比如,平时我们可以使用Math.max方法对数组求出最大值:

var arrayOfValue = [33, 2, 9];
var maxValueFromArray = Math.max.apply(null, arrayOfValue)

这个思路利用了apply接受一个数组作为函数参数的特性。
使用展开符,我们就可以:

var arrayOfValue = [33, 2, 9];
var maxValueFromArray = Math.max(...arrayOfValue);

同理,我们可以这样扩充一个数组:

let values = [2, 3, 4];
let verbose = [1, ...values, 5];

当然,以上都是对数组的展开。

对对象属性的展开符的使用,也已经到了Stage3阶段。今后,我们可以这样写代码:

let warriors = {Steph: 95, Klay: 82, Draymond: 79};
let newWarriors = {
    ...warriors,
    Kevin: 97
}

而不必再使用Object.assign进行对象的扩展。

箭头函数Arrow Function

箭头函数带来的好处无疑是对this的绑定。
现在再回到我们的_handleCommentSubmit函数。做完了初始化工作,我们需要将新的comment通过ajax,异步发送给后端。在成功的回调函数中,进行setState处理:

$.ajax({
    url: this.props.url,
    data: comment,
    success: function(resJson) {
        this.setState({comments: resJson})
    }.bind(this)
})

有经验的同学注意到了我使用bind更改this指向的处理。
在ES6箭头函数下,我们可以直接:

$.ajax({
    url: this.props.url,
    data: comment,
    success: (resJson) => {
        this.setState({comments: resJson})
    }
})

我们再展开看看箭头函数的几种用法(使用方式):

  • 匿名函数,单参数情况,比如实现一个数组的各项平方:
let squares = [1, 2, 3].map(value => value * value);
  • 匿名函数,多参数情况,比如实现一个数组的各项累加:
let sum = [9, 8, 7].reduce((prev, value) => prev + value, 0)

这时候,需要把参数放在括号之中。

  • 非匿名函数,无返回值情况:
const alertUser = (message) => {
    alert(message)
}

这时候,函数体用花括号围起来。

  • 更复杂的情况,比如上面我们用到过的:
const MyComponent = ({children, style}) => (
    <div style={style}>{children}</div>
)

Promises

上面的代码我们使用了jquery的ajax方法发送异步请求,这可能在某些情况下出现“回调地狱”的情况。为此,我们使用基于Promise的fetch方法,进行重构:

_handleCommentSubmit(comment) {
    // ...
    fetch(this.props.url, {
        method: 'POST',
        body: Json.stringify(comment)
    })
        .then((res)=>res.json())
        .then((resJson)=>{
            this.setState({comments: resJson})
        })
        .catch((ex)=>{
            console.error(this.props.url, ex)
        })
}

当然,我们可以把上述代码做的更抽象:

const fetchJson = (path, options) => {
    fetch(`${DOMAIN}${path}`, options)
        .then((res)=>res.json())
}

我们在拉取评论时,就可以:

const fetchComments = () => fetchJson('api/comments')

基于promises的fetch,让异步请求变的灵活可靠。

同时,我再安利一个基于promises的sleep函数

const sleep = (delay = 0) => {
    new Promise((resolve)=>{
        setTimeout(resolve, delay)
    })
}

sleep(3000)
    .then(()=>getUniqueCommentAuthors())
    .then((uniqueAuthors)=>{this.state({uniqueAuthors})})

回到我们的Project中,在后端nodeJS实现里,我们把所有的评论存在data/comments.json文件当中。
在渲染视图时,就不可避免地要进行对data/comments.json文件读取,这个过程也应该是异步完成的:

const readFile = (filePath) => (
    new Promise((resolve, reject)=>{
        fs.readFile(filePath, (err, data)=>{
            if (err) {reject(err)}
            resolve(data)
        })
    })
)

readFile('data/comments.json')
    .then((data)=>console.log('Here is the data', data))
    .catch((ex)=>console.log('Arg!', ex))

Async Functions

当然,Promises实现也有缺点。

更先进的方式,就是使用Async,回到我们的_handleCommentSubmit方法,我们可以重构为:

async _handleCommentSubmit(comment) {
    try {
        let res = await fetch(this.props.url, {
            method: 'POST',
            body: JSON.stringify(comment)
        });
        newComments = await res.json();
    }
    catch (ex) {
        console.error(this.props.url, ex);
        newComments = comments;
    }

    this.setState({comments: newComments});
}

总结

这篇演讲生动形象地阐释了React是怎样与ES next融合天衣无缝的。不论是React也好,还是ES也好,其实笔者认为说到底,都是为了更大限度地解放生产力。

欢迎读者与我交流,有任何问题可以留言。今后几天,将有更多新鲜的react conf视频翻译奉献给大家。

Happy Coding!

PS: 作者Github仓库,欢迎通过代码各种形式交流。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容