资讯专栏INFORMATION COLUMN

学习React系列1-React-tutorial全解析

SnaiLiu / 1084人阅读

摘要:例子全解析近些时间一直在关注,关于如何学习可以参照链接的文章自行制定计划。千里之行,始于足下。此外,输出的内容要解析为,而在默认情况下,基于预防攻击的考虑,对输出的内容是不解析为的。

React-tutorial例子全解析

Talk is cheap,Show me the code

近些时间一直在关注React,关于如何学习React可以参照链接的文章自行制定计划。千里之行,始于足下。本文是React官方的教程上的一个例子,通过详细地学习,从中收获不少,特此做了笔记,与大家共享交流进步。

起步

下载例子,然后进行解压

由于采用的node环境,因此下载解压之后,只需在所在目录运行

</>复制代码

  1. npm install
  2. node server.js

采用默认端口设置,只需打开浏览器,访问http://localhost:3000/

目录结构说明

react-tutorial

</>复制代码

  1. --node_modules
  2. --body-parser:express中间件,用于接收和解析json数据
  3. --express:express框架
  4. --public
  5. --css
  6. --base.css:基本样式文件
  7. --scripts
  8. -- example.jsReact应用js文件
  9. index.html:基本的HTML结构
  10. --.editorconfig:用于在不同的编辑器中统一编辑风格(文件编码)的配置文件
  11. --.gitignoregit相关配置文件
  12. --app.jsonweb app的相关信息
  13. --comments.json:上传的评论数据
  14. --LICENSE:项目代码使用协议
  15. --package.json:项目所依赖的包,npm install的安装包的配置文件
  16. --README.md:项目说明书,里面有使用说明
  17. --requirements.txt:不清楚
  18. --server.js:服务器端的js代码
App功能

此项目构建了一个简单的应用,如图所示

服务器端

服务器端的功能还是相对简单的,通过代码注释的形式来分析

导入了依赖的模块

</>复制代码

  1. var fs = require("fs"); //读写文件
  2. var path = require("path"); //路径
  3. var express = require("express"); //express框架
  4. var bodyParser = require("body-parser"); //中间件

生成app,并且进行配置

</>复制代码

  1. //获取comments.json文件的路径
  2. var COMMENTS_FILE = path.join(__dirname, "comments.json");
  3. //设置端口
  4. app.set("port", (process.env.PORT || 3000));
  5. //设置静态文件的文件目录路径
  6. app.use("/", express.static(path.join(__dirname, "public")));
  7. //启用bodyParser中间件接收请求,并且接收并解析json数据
  8. app.use(bodyParser.json());
  9. app.use(bodyParser.urlencoded({extended: true}));

设置响应头部信息

</>复制代码

  1. app.use(function(req, res, next) {
  2. //允许跨域 CORS
  3. res.setHeader("Access-Control-Allow-Origin", "*");
  4. //缓存设置
  5. res.setHeader("Cache-Control", "no-cache");
  6. next();
  7. });

设置get请求url对应的处理函数(获取评论json数据)

</>复制代码

  1. app.get("/api/comments", function(req, res) {
  2. //读取comments.json文件,并且解析为json数据
  3. fs.readFile(COMMENTS_FILE, function(err, data) {
  4. if (err) {
  5. console.error(err);
  6. process.exit(1);
  7. }
  8. //读取成功后,返回
  9. res.json(JSON.parse(data));
  10. });
  11. });

设置post请求url对应的处理函数(提交评论数据)

</>复制代码

  1. app.post("/api/comments", function(req, res) {
  2. //先读取comments.json文件
  3. fs.readFile(COMMENTS_FILE, function(err, data) {
  4. if (err) {
  5. console.error(err);
  6. process.exit(1);
  7. }
  8. //将文件内容解析为json数据
  9. var comments = JSON.parse(data);
  10. //获取新评论
  11. var newComment = {
  12. id: Date.now(),
  13. author: req.body.author,
  14. text: req.body.text,
  15. };
  16. //添加json数组中
  17. comments.push(newComment);
  18. //将json数据写回到comments.json文件中,并且返回全部的评论数据
  19. fs.writeFile(COMMENTS_FILE, JSON.stringify(comments, null, 4), function(err) {
  20. if (err) {
  21. console.error(err);
  22. process.exit(1);
  23. }
  24. res.json(comments);
  25. });
  26. });
  27. });

启动,监听端口

</>复制代码

  1. app.listen(app.get("port"), function() {
  2. console.log("Server started: http://localhost:" + app.get("port") + "/");
  3. });

web端

web端核心在于example.js文件,结合官网的资料,我们对这个应用进行分析,学习如何构建一个简单的react应用。

组件结构

React践行了Web Components的理念,依照组件化的开发方式,我们来分析这个应用的组件结构(如图所示):

即是:

</>复制代码

  1. -- CommentBox
  2. -- CommentList
  3. -- Comment
  4. -- CommentForm

组件之间的关系图为:

组件Comment

如上述的结构图,我们从最底层开始编写组件Comment,这个组件需要做两件事情

接收上层组件CommentList传递的数据,动态渲染虚拟DOM节点,则从props中读取数据

</>复制代码

  1. //评论人
  2. {this.props.author}
  3. //评论的内容
  4. {this.props.children}

由于评论是支持MarkDown语法的,因此需要使用第三放库marked对用户输入的内容进行处理。

</>复制代码

  1. var rawMarkup = marked(this.props.children.toString(), {sanitize: true});

此外,输出的内容要解析为HTML,而在默认情况下,基于预防XSS攻击的考虑,React对输出的内容是不解析为HTML的。此时,需要利用到特殊的属性dangerouslySetInnerHTML,要将内容放到一个对象的_html属性中,然后将这个对象赋值给dangerouslySetInnerHTML属性

</>复制代码

  1. var html = {_html:"输出的html内容"};

</>复制代码

  1. var Comment = React.createClass({
  2. rawMarkup : function() {
  3. var rawMarkup = marked(this.props.children.toString(),{sanitize:true});
  4. return {_html : rawMarkup}; //React的规则,会读取这个对象的_html内容,
  5. },
  6. render : function() {
  7. return (
  8. {this.props.author}
  9. );
  10. }
  11. });
组件CommentList

组件CommentList需要做的就是接收上一层组件CommentBox传递过来的数据,然后根据数据生成多个子组件Comment

</>复制代码

  1. var CommentList = React.createClass({
  2. render : function() {
  3. var commentNodes = this.props.data.map(function(comment){
  4. return (
  5. {comment.text}
  6. );
  7. });
  8. return (
  9. {commentNodes}
  10. );
  11. }
  12. })

在生成子组件Comment时,将每个子组件的key属性设置为comment.id,这是因为key是一个可选的唯一标识符,通过它可以给组件设置一个独一无二的键,并确保它在一个渲染周期中保持一致,使得React能够更加智能地决定应该重用一个组件,还是销毁并重新创建一个组件,进而提升渲染性能。

组件CommentForm

组件CommentForm需要做的就是两件事情

管理自身的状态this.state(即表单中输入的评论人和评论内容)

当表单输入发生变化时

当表单提交时

当submit事件触发时,调用上一层组件CommentBox的事件处理函数,改变组件CommentBox的状态。

</>复制代码

  1. var CommentForm = React.createClass({
  2. getInitialState : function() {
  3. //设置初始状态,
  4. return {author:"",text:""};
  5. },
  6. handleAuthorChange : function(e) {
  7. this.setState({
  8. author : e.target.value
  9. });
  10. },
  11. handleTextChange : function(e) {
  12. this.setState({
  13. text : e.target.value
  14. });
  15. },
  16. handleSubmit : function(e) {
  17. e.preventDefault();
  18. var author = this.state.author.trim();
  19. var text = this.state.text.trim();
  20. if(!text || !author){ //为空验证
  21. return;
  22. }
  23. //触发评论提交事件,改变父组件的状态
  24. this.props.onCommentSubmit({author:author,text:text});
  25. //改变自身的状态
  26. this.setState({author:"",text:""});
  27. }
  28. });

在这里有一个值得注意的点,那就是抽象的自定义事件commentSubmit和真实的事件submit之间的联系,这是一个相当实用的技巧,在接下来的章节可以看到是如何实现的。

组件CommentBox

作为整个应用的顶层组件,CommentBox需要做的事情有:

从服务器端请求已有的评论数据

将新的评论数据上传到服务器

管理自身的状态,根据状态对视图进行渲染(状态改变的示意图如下)

</>复制代码

  1. var CommentBox = React.createClass({
  2. getInitialState : function(){
  3. return {data : []};
  4. },
  5. loadCommentsFromServer : function() {
  6. //使用了jQuery的Ajax
  7. $.ajax({
  8. url : this.props.url,
  9. dataType : "json",
  10. cache : false,
  11. success : function(data) {
  12. this.setState({data:data});
  13. }.bind(this),
  14. error : function(xhr,status,err){
  15. console.err(this.props.url,status,err.toString());
  16. }.bind(this)
  17. });
  18. },
  19. componentDidMount : function() {
  20. /*
  21. 这个方法属于React组件生命周期方法,在render方法成功调用并且真实的DOM
  22. 已经渲染之后,调用此方法,这个方法发送json数据请求,并且设置一个定时器
  23. ,每隔一段时间就向服务器请求数据
  24. */
  25. this.loadCommentsFromServer();
  26. setInterval(this.loadCommentsFromServer,this.props.pollInterval);
  27. },
  28. handleCommentSubmit : function(comment) {
  29. /*
  30. 这个方法也是比较有意思:
  31. 1. 自定义了一个commentSubmit事件,并且此方法作为该事件的处理函数。
  32. 2. 此方法是在子组件CommentForm的submit事件处理函数中调用
  33. */
  34. var comments = this.state.data;
  35. comment.id = Date.now();
  36. var newComments = comments.concat([comment]);
  37. //改变自身状态
  38. this.setState({data:newComments});
  39. $.ajax({
  40. url : this.props.url,
  41. dataType: "json",
  42. type : "POST",
  43. data : comment,
  44. success : function(data) {
  45. this.setState({data:data});
  46. }.bind(this),
  47. error : function(xhr,status,err) {
  48. //还原数据
  49. this.setState({data:comments});
  50. console.err(this.props.url,status,err.toString());
  51. }.bind(this)
  52. });
  53. },
  54. render : function() {
  55. return (
  56. Comments

  57. );
  58. }
  59. });

最后,只需将组件CommentBox挂载到真实的DOM节点上,就可以看到效果了

</>复制代码

  1. ReactDOM.render(
  2. ,
  3. document.getElementById("content")
  4. );

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/79605.html

相关文章

  • JavaScript - 收藏集 - 掘金

    摘要:插件开发前端掘金作者原文地址译者插件是为应用添加全局功能的一种强大而且简单的方式。提供了与使用掌控异步前端掘金教你使用在行代码内优雅的实现文件分片断点续传。 Vue.js 插件开发 - 前端 - 掘金作者:Joshua Bemenderfer原文地址: creating-custom-plugins译者:jeneser Vue.js插件是为应用添加全局功能的一种强大而且简单的方式。插....

    izhuhaodev 评论0 收藏0
  • React 和 ES6 工作流之 Webpack的使用(第六部分)

    摘要:在上面的列表中,是自解释型的。我们将使用后者。调整文件的内容到这一步,这个应用就具备热刷新的功能。下一步,更新文件中的到现在为止,如果你在控制台运行压缩文件将被创建并且放在路径下面。 这是React和ECMAScript2015系列文章的最后一篇,我们将继续探索React 和 Webpack的使用。 下面是所有系列文章章节的链接: React 、 ES6 - 介绍(第一部分) Rea...

    Mr_houzi 评论0 收藏0

发表评论

0条评论

SnaiLiu

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<