React中Ref 的使用方法详解


Posted in Javascript onApril 28, 2020

本文实例讲述了React中Ref 的使用方法。分享给大家供大家参考,具体如下:

React中Ref 的使用 React v16.6.3

在典型的React数据流中,props是父组件与其子组件交互的唯一方式。要修改子项,请使用new props 重新呈现它。但是,在某些情况下,需要在典型数据流之外强制修改子项。要修改的子项可以是React组件的实例,也可以是DOM元素。对于这两种情况,React都提供了api。

何时使用refs

refs有一些很好的用例:

  • 1.文本选择或媒体播放。
  • 2.触发势在必行的动画。
  • 3.与第三方DOM库集成。

避免将refs用于可以声明性地完成的任何操作。

*不要过度使用Refs

旧版API:字符串引用

如果您之前使用过React,那么您可能熟悉一个旧的API,其中ref属性是一个字符串"textInput",并且DOM节点被访问为this.refs.textInput。建议不要使用它,因为字符串引用有一些问题,被认为是遗留问题,很可能会在未来的某个版本中删除。

回调引用

当组件安装时,React将使用DOM元素调用ref回调,并在卸载时调用null。

在componentDidMount或componentDidUpdate触发之前,Refs保证是最新的.

class CustomTextInput extends React.Component {
 constructor(props) {
 super(props);

 this.textInput = null;

 this.setTextInputRef = element => {
  this.textInput = element;
 };

 this.focusTextInput = () => {
  // Focus the text input using the raw DOM API
  if (this.textInput) this.textInput.focus();
 };
 }

 componentDidMount() {
 // autofocus the input on mount
 this.focusTextInput();
 }

 render() {
 // Use the `ref` callback to store a reference to the text input DOM
 // element in an instance field (for example, this.textInput).
 return (
  <div>
  <input
   type="text"
   ref={this.setTextInputRef}
  />
  <input
   type="button"
   value="Focus the text input"
   onClick={this.focusTextInput}
  />
  </div>
 );
 }
}

refs例子--点击获取input焦点

class Example extends React.Component {
 handleClick() {
 // 使用原生的 DOM API 获取焦点
 this.refs.myInput.focus
 ();
 }
 render() {
 // 当组件插入到 DOM 后,ref 属性添加一个组件的引用于到 this.refs
 return (
  <div>
  <input type="text" ref="myInput" />
  <input
   type="button"
   value="点我输入框获取焦点"
   onClick={this.handleClick.bind(this)}
  />
  </div>
 );
 }
}

使用React.createRef()

React.createRef()React 16.3中引入的API。如果您使用的是早期版本的React,我们建议您使用回调引用。

创建React.createRef()

Refs是使用属性创建的,React.createRef()并通过ref属性附加到React元素。在构造组件时,通常将Refs分配给实例属性,以便可以在整个组件中引用它们。

class MyComponent extends React.Component {
 constructor(props) {
 super(props);
 this.myRef = React.createRef();
 }
 render() {
 return <div ref={this.myRef} />;
 }
}

访问ref

当ref被传递给元素时render,对该节点的引用变得可以在currentref 的属性处访问

const node = this.myRef.current;

ref的值根据节点的类型而有所不同

  • 当在refHTML元素上使用该属性时,ref在构造函数中创建的属性将React.createRef()接收底层DOM元素作为其current属性。
  • 在ref自定义类组件上使用该属性时,该ref对象将接收组件的已安装实例作为其current。

您可能无法ref在函数组件上使用该属性,因为它们没有实例。

class CustomTextInput extends React.Component {
 constructor(props) {
 super(props);
 // create a ref to store the textInput DOM element
 this.textInput = React.createRef();
 this.focusTextInput = this.focusTextInput.bind(this);
 }

 focusTextInput() {
 // Explicitly focus the text input using the raw DOM API
 // Note: we're accessing "current" to get the DOM node
 this.textInput.current.focus();
 }

 render() {
 // tell React that we want to associate the <input> ref
 // with the `textInput` that we created in the constructor
 return (
  <div>
  <input
   type="text"
   ref={this.textInput} />

  <input
   type="button"
   value="Focus the text input"
   onClick={this.focusTextInput}
  />
  </div>
 );
 }
}

current当组件安装时,React将为该属性分配DOM元素,并null在卸载时将其分配回。ref更新发生之前componentDidMount或componentDidUpdate生命周期方法。

无法在函数组件上使用ref属性

function MyFunctionComponent() {
 return <input />;
}

class Parent extends React.Component {
 constructor(props) {
 super(props);
 this.textInput = React.createRef();
 }
 render() {
 // This will *not* work!
 return (
  <MyFunctionComponent ref={this.textInput} />
 );
 }
}

**如果需要引用它,则应该将组件转换为类,就像您需要生命周期方法或状态时一样。
但是,只要引用DOM元素或类组件,就可以在函数组件中使用该ref属性:**

function CustomTextInput(props) {
 // textInput must be declared here so the ref can refer to it
 let textInput = React.createRef();

 function handleClick() {
 textInput.current.focus();
 }

 return (
 <div>
  <input
  type="text"
  ref={textInput} />

  <input
  type="button"
  value="Focus the text input"
  onClick={handleClick}
  />
 </div>
 );
}

将DOM引用公开给父组件

在极少数情况下,可能希望从父组件访问子节点的DOM节点。通常不建议这样做,因为它会破坏组件封装,但它偶尔可用于触发焦点或测量子DOM节点的大小或位置。

虽然可以向子组件添加引用,但这不是一个理想的解决方案,因为只能获得组件实例而不是DOM节点。此外,这不适用于功能组件。

如果使用React 16.3或更高版本,我们建议在这些情况下使用ref forwarding。引用转发允许组件选择将任何子组件的引用公开为自己的组件。可以在ref转发文档中找到有关如何将子DOM节点公开给父组件的详细示例。

如果您使用React 16.2或更低版本,或者您需要比ref转发提供的更多灵活性,您可以使用此替代方法并明确地将ref作为不同名称的prop传递。

如果可能,建议不要暴露DOM节点,但它可以是一个有用的逃生舱。请注意,此方法要求向子组件添加一些代码。如果您完全无法控制子组件实现,则最后一个选项是使用findDOMNode(),但不鼓励使用它StrictMode。

希望本文所述对大家react程序设计有所帮助。

Javascript 相关文章推荐
Javascript 更新 JavaScript 数组的 uniq 方法
Jan 23 Javascript
Javascript 二维数组
Nov 26 Javascript
Javascript变量的作用域和作用域链详解
Apr 02 Javascript
ros::spin() 和 ros::spinOnce()函数的区别及详解
Oct 01 Javascript
谈谈因Vue.js引发关于getter和setter的思考
Dec 02 Javascript
javascript实现一个网页加载进度loading
Jan 04 Javascript
vue-cli配置环境变量的方法
Jul 09 Javascript
React 组件中的 bind(this)示例代码
Sep 16 Javascript
vue 移动端注入骨架屏的配置方法
Jun 25 Javascript
Vue  webpack 项目自动打包压缩成zip文件的方法
Jul 24 Javascript
JS删除数组指定值常用方法详解
Jun 04 Javascript
JS原生实现轮播图的几种方法
Mar 23 Javascript
在Webpack中用url-loader处理图片和字体的问题
Apr 28 #Javascript
react PropTypes校验传递的值操作示例
Apr 28 #Javascript
vue 百度地图(vue-baidu-map)绘制方向箭头折线实例代码详解
Apr 28 #Javascript
Vue + Scss 动态切换主题颜色实现换肤的示例代码
Apr 27 #Javascript
浅析vue cli3 封装Svgicon组件正确姿势(推荐)
Apr 27 #Javascript
react组件基本用法示例小结
Apr 27 #Javascript
react基本安装与测试示例
Apr 27 #Javascript
You might like
其他功能
2006/10/09 PHP
第十四节--命名空间
2006/11/16 PHP
解析PHP可变函数的经典用法
2013/06/20 PHP
php实现的返回数据格式化类实例
2014/09/22 PHP
PHP使用内置dir类实现目录遍历删除
2015/03/31 PHP
PHP滚动日志的代码实现
2015/06/10 PHP
php使用PDO下exec()函数查询执行后受影响行数的方法
2017/03/28 PHP
jQuery 相关控件的事件操作分解
2009/08/03 Javascript
jquery slibings选取同级其他元素的实现代码
2013/11/15 Javascript
jQuery scroll事件实现监控滚动条分页示例
2014/04/04 Javascript
谈谈JavaScript类型系统之Math
2016/01/06 Javascript
jQuery动画效果图片轮播特效
2016/01/12 Javascript
JS/jQuery判断DOM节点是否存在的简单方法
2016/11/24 Javascript
javascript中BOM基础知识总结
2017/02/14 Javascript
Vue.js 实现微信公众号菜单编辑器功能(一)
2018/05/08 Javascript
浅谈Vue render函数在ElementUi中的应用
2018/09/06 Javascript
vue实现div拖拽互换位置
2020/07/29 Javascript
详解mpvue中使用vant时需要注意的onChange事件的坑
2019/05/16 Javascript
js 根据对象数组中的属性进行排序实现代码
2019/09/12 Javascript
es6函数之箭头函数用法实例详解
2020/04/25 Javascript
vue商城中商品“筛选器”功能的实现代码
2020/07/01 Javascript
原生微信小程序开发中 redux 的使用详解
2021/02/18 Javascript
python实现颜色空间转换程序(Tkinter)
2015/12/31 Python
Python如何快速实现分布式任务
2017/07/06 Python
Django 2.0版本的新特性抢先看!
2018/01/05 Python
Django实战之用户认证(用户登录与注销)
2018/07/16 Python
对Python3.6 IDLE常用快捷键介绍
2018/07/16 Python
解决Python内层for循环如何break出外层的循环的问题
2019/06/24 Python
Python Pivot table透视表使用方法解析
2020/09/11 Python
Python jieba库分词模式实例用法
2021/01/13 Python
商场中秋节广播稿
2014/01/17 职场文书
大型晚会策划方案
2014/02/06 职场文书
勾股定理课后反思
2014/04/26 职场文书
社区活动策划方案
2014/08/21 职场文书
镇副书记专题民主生活会对照检查材料思想汇报
2014/10/02 职场文书
党性修养心得体会2016
2016/01/21 职场文书