原文 How Does React Tell a Class from a Function?

译注:

一分钟概览——

React最后采用了在React.Component上加入isReactComponent标识作为区分。

1.在这之前,考虑了ES6的区分方法,但是由于Babel的存在,这个方法不可用。

2.总是调用new,对于一些纯函数组件不适用。而且对箭头函数使用new会出错。

3.把问题约束到React组件下,通过判定原型链来做,但是可能有多个React实例导致判定出错,所以在原型上添加了标识位,标识位是一个对象,因为早期Jest会忽略普通类型如Boolean型。

4.API检测也是可行的,但是API的发展无法预测,每个检测都会带来额外的损耗,所以不是主要做法,但是在现在版本里已经加入了render检测,用来检测prototype.render存在,但是prototype.isReactComponent不存在的场景,这样会抛出一个warning。

以下正文。

思考一下下面这个使用function定义的Greeting组件:

function Greeting() {
return Hello;
}

React也支持class定义:

class Greeting extends React.Component {
render() {
return Hello;
}
}

(直到最近,这是唯一可以使用类似state这种功能的方法。)

当你在使用<Greeting />组件时,其实并不关心它是怎么定义的。

// Class or function — whatever.

但是React自己是关心这些不同的!

如果Greeting是一个函数,React需要去调用它:

// Your code
function Greeting() {
return Hello;
} // Inside React
const result = Greeting(props); // Hello

但是如果Greeting是类,React就需要用new关键字去实例化一个对象,然后立刻调用它的render方法。

// Your code
class Greeting extends React.Component {
render() {
return Hello;
}
} // Inside React
const instance = new Greeting(props); // Greeting {}
const result = instance.render(); // Hello

React有一个相同的目的——得到一个渲染完毕的node(在这个例子里,<p>Hello</p>)。但是如果定义Greeting决定了剩下的步骤。

所以React是如何知道一个组件是类还是函数?

就像我之前的博客,你不需要知道这个东西对于React而言的效果。我同样好几点不了解这些。请不要把这个问题变成一个面试题。事实上,比起React,这篇博客更关注于JavaScript。

这篇博客写给那些富有好奇心的读者,他们想知道为什么React能以一种确定的方式工作。你是这样的人吗?一起深入探讨吧!

这是一段漫长的旅程。这篇博客不会写很多关于React的东西,但是会一掠JavaScript本身的风采,诸如:new,this,class,箭头函数,prototype,__proto__,instanceof,以及这些东西如何在JavaScript中合作。幸运的是,在你使用React的时候,你不必想太多这些事。


首先,我们需要明白为什么区分函数和类如此重要。注意我们怎么使用new操作符去调用一个类:

// If Greeting is a function
const result = Greeting(props); // Hello // If Greeting is a class
const instance = new Greeting(props); // Greeting {}
const result = instance.render(); // Hello

先来对new操作符做了什么给出一个粗浅的定义。


以前,JavaScript没有类的概念。然而,你也可以用纯函数去描述一种近似于类的模式。具体而言,你可以在调用函数之前,添加new,就可以使用任何类似于类构造器的函数了。

// Just a function
function Person(name) {
this.name = name;
} var fred = new Person('Fred'); // ✅ Person {name: 'Fred'}
var george = Person('George'); //

[译]React如何区别class和function的更多相关文章

  1. vuejs与angularjs以及react的区别?

    1.与AngularJS的区别 相同点: 都支持指令:内置指令和自定义指令. 都支持过滤器:内置过滤器和自定义过滤器. 都支持双向数据绑定. 都不支持低端浏览器. 不同点: 1.AngularJS的学 ...

  2. 为什么React事件处理函数必须使用Function.bind()绑定this?

    最近在React官网学习Handling Events这一章时,有一处不是很明白.代码如下: class Toggle extends React.Component { constructor(pr ...

  3. [译] React 16.3(.0-alpha) 新特性

    原文地址:What's new in React 16.3(.0-alpha) 原文作者:Bartosz Szczeciński 译文出自:掘金翻译计划 本文永久链接:github.com/xitu/ ...

  4. [译]React Context

    欢迎各位指导与讨论 : ) 前言 由于笔者英语和技术水平有限,有不足的地方恳请各位指出.我会及时修正的 O(∩_∩)O 当前React版本 15.0.1 时间 2016/4/25 正文 React一个 ...

  5. [译]React 在服务端渲染的实现

    原文地址:Server-Side React Rendering 原文作者:Roger Jin React 在服务端渲染的实现 React是最受欢迎的客户端 JavaScript 框架,但你知道吗(可 ...

  6. 前端框架 vue 和 react 的区别

    前言:最近需要使用 react,以前用过 vue,故来总结两者的区别. 首先React与vue有几点相同之处 1.都使用了Virtual DOM 2.提供了响应式(Reactive)和组件化(Comp ...

  7. [React] Preventing extra re-rendering with function component by using React.memo and useCallback

    Got the idea form this lesson. Not sure whether it is the ncessary, no othere better way to handle i ...

  8. JS中的函数声明和函数表达式的区别,即function(){}和var function(){},以及变量提升、作用域和作用域链

    一.前言 Uncaught TypeError: ... is not a function function max(){}表示函数声明,可以放在代码的任何位置,也可以在任何地方成功调用: var ...

  9. Vue和React的区别,以及如何选择?

    简介 React:React是一个用于创建可重用且有吸引力的UI组件的库.它非常适合代表经常变化的数据的组件. Vue:Vue.js是一个开源JavaScript框架,能够开发单页面应用程序.它还可以 ...

随机推荐

  1. C# WCF服务入门

    之前在公司用的服务端是wcf写的,但是没有深入研究,最近找工作,面试的时候好多人看到这个总提问,这里做个复习 就用微软官方上的例子,搭一个简单的wcf服务,分6步 1 定义服务协定也就是契约,其实就是 ...

  2. Spring AOP注解通过@Autowired,@Resource,@Qualifier,@PostConstruct,@PreDestroy注入属性的

    本文介绍了使用spring注解注入属性的方法. 使用注解以前,注入属性通过类以及配置文件来实现.现在,注入属性可以通过引入@Autowired注解,或者@Resource,@Qualifier,@Po ...

  3. CentOS安装软件出现错误:bash: /usr/local/bin/rar: /lib/ld-linux.so.2: bad ELF interpreter: No such file or directory

    CentOS安装软件出现错误: bash: /usr/local/bin/rar: /lib/ld-linux.so.2: bad ELF interpreter: No such file or d ...

  4. 如何优雅地使用Markdown (Sublime 3 + MarkdownEditing+OmniMarkupPreviewer)

    最近开始上手Sublime 3 作为Markdown 的重度使用者自然关于Markdown的插件是必不可少的 . 在这里记录分享一下我常用的两款Markdown插件. MarkdownEditing ...

  5. andriod导入v4包导致的错误

    最近升级android studio到版本3.0.1后,想要使用FragmentActivity这个类,导入v4包,发现R文件报错了,也就是找不到的意思. 如图:导包 此时选中v4包导进去. 确定之后 ...

  6. Mysql 性能调优参数

    1.DEFAULT_STORAGE_ENGINE <br\>如果你已经在用MySQL 5.6或者5.7,并且你的数据表都是InnoDB,那么表示你已经设置好了.如果没有,确保把你的表转换为 ...

  7. CSS animation online生成工具

    利用HTML5.css的一些动画功能,可以设计出非常炫酷的动画,但是由于并不是所有的浏览器都支持,所以可能需要prefix,这个过程是比较烦的.一个比较好用的线上工具: http://matthewl ...

  8. Oracle从入门到精通 限定查询和排序查询的问题

    视频课程:李兴华 Oracle从入门到精通视频课程 学习者:阳光罗诺 视频来源:51CTO学院 知识点 SQL语句的执行顺序 限定符号的使用.   具体内容: 如果想要对所选择的数据进行控制,就可以使 ...

  9. coder/programmer engineer Chirf Technology Offcer

    大概是某个C轮融资的医疗网站CTO被离职.而CTO是一个知乎大V和微信大号.此事一出,在微信群有支持也有反对之声.支持此CTO被离职的认为其在工作时没有Review程序,自己不写代码,而是热衷出没于技 ...

  10. Service Discovery in WCF 4.0 – Part 2 z

    Service Discovery in WCF 4.0 – Part 2 In the previous post I discussed about the basic usage of WCF ...