ref 的功能,在 react 当中。我们写了一个组件,在这个时候,我们的 render function 里面我们会渲染一系列的子组件或者 dom 节点,有时候我们会希望有这样的需求,就是我们要获取某个 dom 节点,或者是某个子组件的实例。去对他进行一些手动的操作,而不仅仅是 props 更新这种方式去更新这个节点。比如我们要获取一个 dom 节点,自己去绑定某一些事件,然后去做一些操作。通过 ref 这个特殊的 attr 非常方便的在组件内部去获取到子节点的具体的一个实例。这就是 ref 的核心功能。
代码
import React from 'react'

export default class RefDemo extends React.Component {
constructor() {
super()
this.objRef = React.createRef()
} componentDidMount() {
setTimeout(() => {
this.refs.stringRef.textContent = 'string ref got'
this.methodRef.textContent = 'method ref got'
this.objRef.current.textContent = 'obj ref got'
}, 1000)
} render() {
return (
<>
<p ref="stringRef">span1</p>
<p ref={ele => (this.methodRef = ele)}>span3</p>
<p ref={this.objRef}>span3</p>
</>
)
}
}
如上,在 react 当中有3中使用 ref 的方式
1、string ref 
<p ref="stringRef">span1</p>
=>
this.refs.stringRef.textContent = 'string ref got'

也就是我们想要获取的这个节点的 props 上面使用一个 ref 属性,然后传入一个字符串,传入这个字符串之后呢,react 在完成这一个节点的渲染之后,他会在 this.refs 这个对象上面挂载这个 string 对应的一个 key 。那么这个 key 他所指向的就是我们这个节点他的实例的对象。如果是 dom 节点,他就对应 dom 的实例,如果是子组件就会是子组件的实例,就是我们的 class component。如果是 function component 怎么办呢,正常来讲,他是会失败的,也就是拿到的会是个 undefined 。因为 function component 没有实例。这是第一种使用的方式,也是 v16 之前用的最多的方式。也是最不被推荐的一种方式。在下一个大版本之中会被废弃的一种方式

2、function 
<p ref={ele => (this.methodRef = ele)}>span2</p>
=>
this.methodRef.textContent = 'method ref got'

这个我们的 ref 传入的是一个方法,这个方法会接收一个参数,这个参数就是这个节点对应的实例,如果是 dom 节点对应的是 dom 的实例,如果是组件对应的是组件的实例。然后我们可以自己去this上面去挂载某一个属性,比如在这里,对应的是 methodRef 。

3、createRef (这是 react 提供给我们的 api)
this.objRef = React.createRef()
这个 api 就是 React.createRef(),这个对象其实就是这样的 { current: null } 。然后这个对象传入给某一个节点
<p ref={this.objRef}>span3</p>

在组件渲染完成之后,会把这个节点对应的实例挂载到这个对象的 current 这个属性里面,我们要调用他就是通过

this.objRef.current.textContent = 'obj ref got'
运行这个js文件,显示 span1 span2 span3 。一秒钟后被 string ref got , method ref got, obj ref got 替换。然后我们看 createRef 这个的源码,因为其他的两种方式并没有在源码上有任何的体现,因为他们传入的一个是 string ,一个 function 。并没有直接挂载到 React 上面的一个 api。只有第三种方式 React.createRef 给我们创建了这么一个东西。
我们看 React.js 里面 createRef 引自 ReactCreateRef
/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
* @flow
*/ import type {RefObject} from 'shared/ReactTypes'; // an immutable object with a single mutable value
export function createRef(): RefObject {
const refObject = {
current: null,
};
if (__DEV__) {
Object.seal(refObject);
}
return refObject;
}

点开源码,震惊了,非常的少。 也非常的简单,他返回的就是这样一个对象 refObject 。他里面 current 是 null 。如果是开发状态下 Object.seal(refObject) 。refObject 这个对象被封闭,阻止添加新属性并将现有属性标记为不可配置。当前属性的值只要可写就可以改变

React源码 React ref的更多相关文章

  1. React源码 React.Component

    React中最重要的就是组件,写的更多的组件都是继承至 React.Component .大部分同学可能都会认为 Component 这个base class 给我们提供了各种各样的功能.他帮助我们去 ...

  2. React源码 React.Children

    children是什么意思呢?就是我们拿到组件内部的props的时候,有props.children这么一个属性,大部分情况下,我们直接把 props.children 渲染到 JSX 里面就可以了. ...

  3. 读react源码准备

    git源码地址:https://github.com/facebook/react react 里面就是 react源码 react里面的react文件夹就是react源码,react源码非常的少,总 ...

  4. React源码解析:ReactElement

    ReactElement算是React源码中比较简单的部分了,直接看源码: var ReactElement = function(type, key, ref, self, source, owne ...

  5. React躬行记(16)——React源码分析

    React可大致分为三部分:Core.Reconciler和Renderer,在阅读源码之前,首先需要搭建测试环境,为了方便起见,本文直接采用了网友搭建好的环境,React版本是16.8.6,与最新版 ...

  6. React源码之组件的实现与首次渲染

    react: v15.0.0 本文讲 组件如何编译 以及 ReactDOM.render 的渲染过程. babel 的编译 babel 将 React JSX 编译成 JavaScript. 在 ba ...

  7. React源码剖析系列 - 生命周期的管理艺术

    目前,前端领域中 React 势头正盛,很少能够深入剖析内部实现机制和原理.本系列文章希望通过剖析 React 源码,理解其内部的实现原理,知其然更要知其所以然. 对于 React,其组件生命周期(C ...

  8. react 源码之setState

    今天看了react源码,仅以记录. 1:monorepo (react 的代码管理方式) 与multirepo 相对. monorepo是单代码仓库, 是把所有相关项目都集中在一个代码仓库中,每个mo ...

  9. React 源码剖析系列 - 不可思议的 react diff

      简单点的重复利用已有的dom和其他REACT性能快的原理. key的作用和虚拟节点 目前,前端领域中 React 势头正盛,使用者众多却少有能够深入剖析内部实现机制和原理. 本系列文章希望通过剖析 ...

随机推荐

  1. ES6-Generator基础用法

    Generator简介: 生成器,本身是函数,执行后返回迭代对象,函数内部要配合yield使用Generator函数会分段执行,遇到yield暂停. 使用Generator注意点:function 和 ...

  2. Affy包 estrogen包

    下载安装 if (!requireNamespace("BiocManager", quietly = TRUE)) install.packages("BiocMana ...

  3. 2018-2019-2 20165315《网络对抗技术》Exp7 网络欺诈防范

    2018-2019-2 20165315<网络对抗技术>Exp7 网络欺诈防范 一.实验内容 本实践的目标理解常用网络欺诈背后的原理,以提高防范意识,并提出具体防范方法.具体实践有 简单应 ...

  4. elasticsearch容量规划

    https://docs.bonsai.io/article/123-capacity-planning Capacity Planning Capacity planning is the proc ...

  5. javascript中的each遍历

    each的用法  1.数组中的each 复制代码 var arr = [ "one", "two", "three", "four ...

  6. Effective.Java第45-55条(规范相关)

    45.  明智谨慎地使用Stream 46.  优先考虑流中无副作用的函数 47.  优先使用Collection而不是Stream作为方法的返回类型 48.  谨慎使用流并行 49.  检查参数有效 ...

  7. 大一C语言课程设计——班级档案管理系统

    记录我在大一第二学期期末做的C语言课程毕业设计 1. 班级档案管理系统运用到的主要结构体 typedef struct birthday //出生日期{int year;int month;int d ...

  8. java -jar参数运行方式设置classpath

    转载自:https://www.cnblogs.com/aggavara/archive/2012/11/16/2773246.html 当用java -jar yourJarExe.jar来运行一个 ...

  9. 50道Java线程面试题分析及答案

    下面是Java线程相关的热门面试题,你可以用它来好好准备面试. 1) 什么是线程?线程是操作系统能够进行运算调度的最小单位,它被包含在进程之中,是进程中的实际运作单位.程序员可以通过它进行多处理器编程 ...

  10. html引入公共模块

    如果没有母版页,那么大量相同布局的页面会有很多相同的代码,那么这就提到了一个概念,叫重用性:可以将相同布局的代码放在一个单独的文件,里面写一些公共模块,那么在其他页面只需要在指定位置引入他们就可以了写 ...