- React Components
- Christopher Pitt
- 855字
- 2021-07-09 19:34:44
Using modern JavaScript
React components are wonderfully encapsulated. Each component is a blueprint for what a focused bit of markup should look like at any moment. They're reusable and can change their behavior depending on the context provided. Does that remind you of another programming paradigm?
Let's talk about JavaScript. JavaScript has a prototypical inheritance model. That means different objects can have a common structure. The structure of one object can be derived from the structure of another.
It also means that changes to the original object are inherited in all derivative objects. Let me illustrate this with some code:
var Page = function(content) { this.content = content; }; Page.prototype.render = function() { return "<p>" + this.content + "</p>"; } var Post = function(tags, content) { this.tags = tags; Page.call(this, content); }; Post.prototype = new Page(); Post.prototype.render = function() { var page = Page.prototype.render.call(this); return "<ul>" + this.renderTags() + "</ul>" + page; }; Post.prototype.renderTags = function() { return "<li>" + this.tags.join("</li></li>") + "</li>"; }; var page = new Page("Welcome to my site!"); var post = new Post(["news"], "A new product!"); Page.prototype.render = function() { return "<section>" + this.content + "</section>"; };
I begin by creating a function called Page
, which requires a content
parameter. A simple render
method returns that content, wrapped in a p
tag. This seems like a good starting point for a website.
Next, I decide to make a second type called Post
. Objects of this type have tags, so I create a new initialization function to store them. I want Post
to behave almost like a Page
type, so I call the Page
initialization function.
To inherit the Page
methods in Post
, I need to link their prototypes. I can then choose to override the render
method and add new methods to the derived type. I can also change the Page
type and these changes will be inherited by objects of the Post
type. The connection happens because a prototype is a reference and not a copy.
Depending on the programming languages you grew up with, prototypical inheritance might be tricky at first. Many new developers learn (incorrectly) that object-oriented code means class-oriented code. Dynamic concepts such as prototypes are foreign to them. In the past, this led to a few libraries implementing "pretend" classes. They created patterns that would make code appear as if it was class-oriented.
Then, ES6 added the class
keyword. It's a formalization of the pattern I just showed you. It's a syntactic shortcut to prototypical inheritance.
We could reduce the previous code to:
class Page { constructor(content) { this.content = content; } render() { return "<p>" + this.content + "</p>"; } } class Post extends Page { constructor(tags, content) { super(content); this.tags = tags; } render() { var page = super.render(); return "<ul>" + this.renderTags() + "</ul>" + page; } renderTags() { return "<li>" + this.tags.join("</li></li>") + "</li>"; } } var page = new Page("Welcome to my site!"); var post = new Post(["news"], "A new product!");
Note
If you're trying to run this using Node (preferably a version greater than 4.1), you may need to add use strict
at the top of the file.
Notice how much clearer things are? If you want to use classes, then this syntactic shortcut is brilliant!
Let's look at a typical ES5-compatible React component:
var Page = React.createClass({ render: function() { return <p>{this.props.content}</p>; } }); var Post = React.createClass({ render: function() { var page = <Page content={this.props.content} /> var tags = this.renderTags(); return <p><ul>{tags}</ul>{page}</p>; }, renderTags: function() { return this.props.tags.map(function(tag, i) { return <li key={i}>{tag}</li>; }); } }); ReactDOM.render( <Post tags={["news"]} content="A new product!" />, document.querySelector(".react") );
You've probably seen this kind of code before. It's called JSX and it's a JavaScript superset language. The idea is that the markup and the supporting logic are created and stored together.
Note
React components must return a single React node, which is why we wrap the tags and page elements in a p
element. If you are using React in the browser, you also need to render your components to an existing DOM node (like I've just rendered the post to .react
).
We'll get into some of the specifics in later chapters, but this is doing pretty much the same thing as before. We create a base component called Page
. It renders a property instead of a constructor parameter.
The Post
component composes the Page
component. This style of React code doesn't support component inheritance. For that, we need ES6 code:
class Page extends React.Component { render() { return <p>{this.props.content}</p>; } } class Post extends Page { render() { var page = super.render(); var tags = this.renderTags(); return <p><ul>{tags}</ul>{page}</p>; } renderTags() { return this.props.tags.map(function(tag, i) { return <li key={i}>{tag}</li>; }); } }
We could still compose Page
within Post
, but that's not the only option with ES6. This code resembles the non-React version we saw earlier.
In upcoming chapters, we'll learn many useful features of ES6 that'll allow us to create modern, expressive React components.
Note
If you want to look ahead a little, check out http://babeljs.io/docs/learn-es2015. It's a great place to learn the main features of ES6!
Babel is the cross-compilation tool we'll use to turn ES6 code into ES5 code:

- CockroachDB權威指南
- ASP.NET MVC4框架揭秘
- 高效微控制器C語言編程
- SQL Server 2016從入門到精通(視頻教學超值版)
- React Native Cookbook
- 算法精粹:經典計算機科學問題的Python實現
- 精通軟件性能測試與LoadRunner實戰(第2版)
- 小程序開發原理與實戰
- AIRIOT物聯網平臺開發框架應用與實戰
- Python 3.7從入門到精通(視頻教學版)
- Learning Material Design
- Getting Started with Polymer
- Visual C++從入門到精通(第2版)
- Shopify Application Development
- 征服C指針(第2版)