react中的refs属性的使用方法

更新日期: 2018-05-18阅读量: 2012标签: react

react中,render执行的结果得到的并不是真正的DOM节点,结果仅仅是轻量级的JavaScript对象,我们称之为virtual DOM。

虚拟DOM是react的一大亮点,具有batching(批处理)和高效的Diff算法。这让我们可以无需担心性能问题而”毫无顾忌”的随时“刷新”整个页面,由虚拟 DOM来确保只对界面上真正变化的部分进行实际的DOM操作。但是,有些场景需要获取某一个真实的DOM元素来交互,比如文本框的聚焦、触发强制动画等。所以怎么获取真实的dom元素呢?嗯、这就是今天的重点啦:refs

React 支持一种非常特殊的属性 Ref ,你可以用来绑定到 render() 输出的任何组件上。这个特殊的属性允许你引用 render() 返回的相应的支撑实例( backing instance )。这样就可以确保在任何时间总是拿到正确的实例。那,具体怎么用,且看下面


1、为DOM元素添加Ref

ref属性接收一个回调函数,这个回调函数在组件挂载或者卸载的时候被调用。当ref用于一个html元素的时候,ref指定的回调函数在调用的时候会接收一个参数,该参数就是指定的DOM元素。

class Input extends Component {  
    constructor(props){  
        super(props);  
       this.focus = this.focus.bind(this)  
    }  
    focus(){  
        this.textInput.focus();  
    }  
      
    render(){  
        return (  
            <div>  
                <input ref={(input) => { this.textInput = input }} />//input参数表示DOM本身  
                <button  onClick={this.focus}>让input获取焦点</button>  
            </div>  
        )  
    }  
}

当我们在DOM Element中使用ref时,回调函数将接收当前的DOM元素作为参数,然后存储一个指向这个DOM元素的引用。那么在示例代码中,我们已经把input元素存储在了this.textInput中,在focus函数中直接使用原生DOM API实现focus聚焦。

2.为组件Component添加Ref

当ref属性用于一个class指定的自定义组件的时候,ref回调函数会接收到一个挂载的组件实例作为参数。

下面的例子展示了当CustomTextInput被挂载后马上模拟被点击:

class AutoFocusTextInput extends Component {  
    componentDidMount(){  
        this.textInput.focus();  
    }  
      
    render(){  
        return (  
            <Input ref={(input) => { this.textInput = input }}>  
        )//ref回调函数会接收一个挂载的组件实例作为参数  
    }  
}

Input组件就是第一个实例的组件哦
当我们在<Input>中添加ref属性时,其回调函数接收已经挂载的组件实例<Input>作为参数,并通过this.textInput访问到其内部的focus方法。也就是说,上面的示例代码实现了当AutoFocusTextInput组件挂载后<Input>组件的自动聚焦。
注意:<Input>组件必须是使用class声明的组件
不能在无状态组件中使用`ref`。原因很简单,因为ref引用的是组件的实例,而无状态组件准确的说是个函数组件(Functional Component),没有实例。

3.为父组件暴露一个DOM的ref属性

这是Facebook非常不推荐的做法,因为这样会打破组件的封装性,这种方法只是某些特殊场景下的权宜之计。我们看看如何实现,上代码:

function CustomTextInput(props) {  
  return (  
    <div>  
     //子级组件接收到父级组件传递过来的inputRef函数  
     //当子级组件实例化的时候会将该input传入到该函数中,此时父级组件会  
     //接收到子级组件的DOM结构(input元素)  
      <input ref={props.inputRef} />  
    </div>  
  );  
}  
class Parent extends React.Component {  
  render() {  
    return (  
      <CustomTextInput  
      //为子级组件传入一个inputRef函数  
        inputRef={el => this.inputElement = el}  
      \/>  
    );  
  }  
}


原理就是父组件把ref的回调函数当做inputRefprops传递给子组件,然后子组件<CustomTextInput>把这个函数和当前的DOM绑定,最终的结果是父组件<Parent>的this.inputElement存储的DOM是子组件<CustomTextInput>中的input。

官方推荐Ref常用情况

第一:管理焦点,文本选择,媒体播放(媒体回放)
第二:触发动画
第三:集成第三方的DOM库


结语:

`refs`提供的是另一种与react传统响应数据流完全不同的组件间交互方式,所以官方指出不要过度使用`refs`,而且从官方对它的态度来看,未来或许有更好的API来取代它。但目前来说`refs`仍是一个不错的解决方案。

来源:https://blog.csdn.net/lhjuejiang/article/details/80341231


站长推荐

1.云服务推荐: 国内主流云服务商,各类云产品的最新活动,优惠券领取。地址:阿里云腾讯云华为云

链接: https://www.fly63.com/article/detial/783

React.memo()、useCallback()、useMemo() 区别及基本使用

React.memo是一个高阶组件(参数为组件,返回值为新组件的函数即为高阶组件),对外部而言,React.memo会检查props的变更,仅当传入的props发生变化时组件才会重新渲染

Context - React跨组件访问数据的利器

Context提供了一种跨组件访问数据的方法。它无需在组件树间逐层传递属性,也可以方便的访问其他组件的数。在经典的React应用中,数据是父组件通过props向子组件传递的

如果没有virtual dom,react会怎样?

如果react火起来的时候,没有virtual dom,会怎样?你还会选择使用react吗?这是一个历史假设问题。但今天来看,这个问题却非常有趣,因为,在经历对react的狂热追捧之后

React Server Component 可能并没有那么香

前段时间 React 团队发布了一项用于解决 React 页面在多接口请求下的性能问题的解决方案 React Server Components。当然该方案目前还在草案阶段,官方也只是发了视频和一个示例 demo 来说明这个草案。

React + Webpack 构建打包优化

React 相关的优化:使用 babel-react-optimize 对 React 代码进行优化,检查没有使用的库,去除 import 引用,按需打包所用的类库,比如 lodash 、echarts 等.Webpack 构建打包存在的问题两个方面:构建速度慢,打包后的文件体积过大

Vue和React有什么区别与优劣?我到底应该选择哪谁?

前端开发框架从最开始的jquery时代,到后来backbone,angular1,再到现在vue和react两分天下,也才用了不到十年的光景。

react 使用 use-immer 更简洁的更新嵌套 state

state 可以保存任何类型的 JavaScript 值,包括对象。但是你不应该直接改变你在 React state 中持有的对象。相反,当您想要更新一个对象时,您需要创建一个新的对象(或者复制一个现有的对象)

解决vscode 开发react 导入绝对路径 无法跳转的问题

相对路径可正常跳转,但是在webpack配置alias使用绝对路径后无法跳转.解决办法:需要添加一个jsconfig文件,如下:

useEffect引起的React Hooks深入了解

在进入正式阅读之前,最好先思考一下下面的问题:React Hooks真的有生命周期吗?React Hooks的函数里面定义的函数或者变量会被缓存吗,这样下次再调用组件的时候就可以不用重新声明了。

在react jsx中,为什么使用箭头函数和bind容易出现问题

因为()=>this.deleteUser(user.id)每执行一次就会生成一个新的函数,当然bind也是这样干的,所以在PureComponent的shallowCompare中认为onDeleteClick的值已经被修改,所以触发了重新渲染。看吧,使用箭头函数和bind会造成性能浪费,作为一个节约的程序员应该避免如此。

点击更多...

内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!