在 React.js 中将 props 传递给父组件

IT技术 javascript reactjs
2021-01-29 02:16:30

props在 React.js 中,是否没有一种简单的方法可以使用事件将子项传递给其父项?

var Child = React.createClass({
  render: function() {
    <a onClick={this.props.onClick}>Click me</a>
  }
});

var Parent = React.createClass({
  onClick: function(event) {
    // event.component.props ?why is this not available?
  },
  render: function() {
    <Child onClick={this.onClick} />
  }
});

我知道您可以使用受控组件来传递输入的值,但最好传递整个 kit n'kaboodle。有时子组件包含一组您不想查找的信息。

也许有一种方法可以将组件绑定到事件?

更新 – 2015 年 9 月 1 日

在使用 React 一年多之后,在 Sebastien Lorber 的回答的激励下,我得出结论,将子组件作为参数传递给父组件中的函数实际上并不是React 的方式,也不是一个好主意。我已经换了答案。

6个回答

编辑:请参阅 ES6 更新示例的最终示例。

这个答案只是处理直接父子关系的情况。当父母和孩子可能有很多中介时,请查看此答案

其他解决方案没有抓住重点

虽然它们仍然可以正常工作,但其他答案缺少一些非常重要的东西。

在 React.js 中,是否没有一种简单的方法可以使用事件将子项的props传递给其父项?

父级已经拥有该子级props!: 如果孩子有一个props,那是因为它的父母把那个props提供给了孩子!为什么你想让孩子把props传回给父母,而父母显然已经有了那个props?

更好的实施

孩子:真的没有必要比这更复杂。

var Child = React.createClass({
  render: function () {
    return <button onClick={this.props.onClick}>{this.props.text}</button>;
  },
});

有一个孩子的父级:使用它传递给子级的值

var Parent = React.createClass({
  getInitialState: function() {
     return {childText: "Click me! (parent prop)"};
  },
  render: function () {
    return (
      <Child onClick={this.handleChildClick} text={this.state.childText}/>
    );
  },
  handleChildClick: function(event) {
     // You can access the prop you pass to the children 
     // because you already have it! 
     // Here you have it in state but it could also be
     //  in props, coming from another parent.
     alert("The Child button text is: " + this.state.childText);
     // You can also access the target of the click here 
     // if you want to do some magic stuff
     alert("The Child HTML is: " + event.target.outerHTML);
  }
});

JsFiddle

带有子项列表的父项:您仍然拥有父项所需的一切,不需要让子项变得更复杂。

var Parent = React.createClass({
  getInitialState: function() {
     return {childrenData: [
         {childText: "Click me 1!", childNumber: 1},
         {childText: "Click me 2!", childNumber: 2}
     ]};
  },
  render: function () {
    var children = this.state.childrenData.map(function(childData,childIndex) {
        return <Child onClick={this.handleChildClick.bind(null,childData)} text={childData.childText}/>;
    }.bind(this));
    return <div>{children}</div>;
  },

  handleChildClick: function(childData,event) {
     alert("The Child button data is: " + childData.childText + " - " + childData.childNumber);
     alert("The Child HTML is: " + event.target.outerHTML);
  }
});

JsFiddle

也可以使用this.handleChildClick.bind(null,childIndex)然后使用this.state.childrenData[childIndex]

请注意,我们正在绑定null上下文,否则 React 会发出与其自动绑定系统相关的警告使用 null 意味着您不想更改函数上下文。另见

关于其他答案中的封装和耦合

在耦合和封装方面,这对我来说是一个主意:

var Parent = React.createClass({
  handleClick: function(childComponent) {
     // using childComponent.props
     // using childComponent.refs.button
     // or anything else using childComponent
  },
  render: function() {
    <Child onClick={this.handleClick} />
  }
});

使用 props:正如我上面解释的,你已经在父组件中拥有了 props,所以传递整个子组件来访问 props 是没有用的。

使用 refs:您已经在事件中拥有点击目标,并且在大多数情况下这已经足够了。此外,您可以直接在孩子上使用 ref :

<Child ref="theChild" .../>

并访问父节点中的 DOM 节点

React.findDOMNode(this.refs.theChild)

对于更高级的情况,您要访问父级中子级的多个引用,子级可以直接在回调中传递所有 dom 节点。

组件有一个接口(props),父组件不应该对子组件的内部工作做任何假设,包括它的内部 DOM 结构或它声明了 refs 的 DOM 节点。使用子项的 ref 的父项意味着您将 2 个组件紧密耦合。

为了说明这个问题,我将引用关于Shadow DOM 的引用,它在浏览器中用于渲染诸如滑块、滚动条、视频播放器之类的东西......:

他们在您(Web 开发人员)可以访问的内容与被视为实现细节的内容之间创建了一个界限,因此您无法访问。然而,浏览器可以随意跨越这个边界。有了这个边界,他们就能够像您一样使用相同的古老 Web 技术构建所有 HTML 元素,无需使用 div 和 span。

问题是,如果让子实现细节泄漏到父中,则很难在不影响父的情况下重构子。这意味着作为库作者(或作为使用 Shadow DOM 的浏览器编辑器)这是非常危险的,因为您让客户端访问太多,这使得在不破坏复古兼容性的情况下升级代码变得非常困难。

如果 Chrome 实现了它的滚动条,让客户端访问该滚动条的内部 dom 节点,这意味着客户端可能有可能简单地破坏该滚动条,并且当​​ Chrome 在重构后执行其自动更新时,应用程序将更容易中断滚动条...相反,它们只允许访问一些安全的东西,比如用 CSS 自定义滚动条的某些部分。

关于使用其他任何东西

在回调中传递整个组件是危险的,并且可能会导致新手开发人员做非常奇怪的事情,例如在父级内部调用childComponent.setState(...)orchildComponent.forceUpdate()或为其分配新变量,从而使整个应用程序更难以推理。


编辑:ES6 示例

由于现在很多人使用 ES6,这里是 ES6 语法的相同示例

孩子可以很简单:

const Child = ({
  onClick, 
  text
}) => (
  <button onClick={onClick}>
    {text}
  </button>
)

父级可以是一个类(它最终可以管理状态本身,但我在这里将它作为props传递:

class Parent1 extends React.Component {
  handleChildClick(childData,event) {
     alert("The Child button data is: " + childData.childText + " - " + childData.childNumber);
     alert("The Child HTML is: " + event.target.outerHTML);
  }
  render() {
    return (
      <div>
        {this.props.childrenData.map(child => (
          <Child
            key={child.childNumber}
            text={child.childText} 
            onClick={e => this.handleChildClick(child,e)}
          />
        ))}
      </div>
    );
  }
}

但是如果不需要管理状态也可以简化:

const Parent2 = ({childrenData}) => (
  <div>
     {childrenData.map(child => (
       <Child
         key={child.childNumber}
         text={child.childText} 
         onClick={e => {
            alert("The Child button data is: " + child.childText + " - " + child.childNumber);
                    alert("The Child HTML is: " + e.target.outerHTML);
         }}
       />
     ))}
  </div>
)

JsFiddle


PERF WARNING(适用于 ES5/ES6):如果你使用PureComponentor shouldComponentUpdate,上面的实现不会默认优化,因为使用onClick={e => doSomething()}, or 直接在渲染阶段绑定,因为它会在每次父级渲染时创建一个新函数。如果这是您的应用程序中的性能瓶颈,您可以将数据传递给子项,并将其重新注入“稳定”回调(在父类上设置,并绑定到this类构造函数),以便PureComponent优化可以启动,或者您可以实现自己的shouldComponentUpdate并忽略props比较检查中的回调。

您还可以使用Recompose库,它提供更高阶的组件来实现微调优化:

// A component that is expensive to render
const ExpensiveComponent = ({ propA, propB }) => {...}

// Optimized version of same component, using shallow comparison of props
// Same effect as React's PureRenderMixin
const OptimizedComponent = pure(ExpensiveComponent)

// Even more optimized: only updates if specific prop keys have changed
const HyperOptimizedComponent = onlyUpdateForKeys(['propA', 'propB'])(ExpensiveComponent)

在这种情况下,您可以使用以下方法优化 Child 组件:

const OptimizedChild = onlyUpdateForKeys(['text'])(Child)
我对您的答案进行了编辑,即关于 refs 的部分。omerwazir.com/posts/react-getdomnode-replaced-with-findDOMNode
2021-03-13 02:16:30
是的,在去年使用 React 之后,我同意——没有充分的理由通过函数传递实例化组件。我将弃用我自己的答案。我也愿意改变答案,也许会做一些修改:你的“更好的实施”没有解决出现这个问题的实际原因,即“我怎么知道我的哪个孩子是从一个群体中选出的?” Flux 是任何体面大小的应用程序的答案,但是,在小型应用程序中,我很高兴通过props. 同意?
2021-03-30 02:16:30
@KendallB 很高兴您同意我的回答 :)How do I know which of my children was chosen among a group?这实际上并不是原始问题的一部分,但我在回答中包含了解决方案。与您的编辑不同,我认为根本不需要修改子项,即使在处理列表时也是如此,因为您可以bind直接在父项中使用。
2021-03-31 02:16:30
我也在努力让它在最新的 React 中工作,需要改变什么?
2021-04-05 02:16:30
嗨@SebastienLorber 我来自未来。关于onClick={this.handleChildClick.bind(null,childData)}你提到的关于自动绑定的这一行,既然 React 没有包含在类模型中,这是否仍然适用我已经阅读了很多东西,但这部分仍然让我感到困惑。class model顺便说一下,我正在使用 ES6 ,所以我改成nullthis,我的代码似乎仍然有效。你会如何将你的那部分代码翻译成 ES6 风格?
2021-04-06 02:16:30

更新(2015 年 9 月 1 日):OP 使这个问题有点像一个移动目标。它又更新了。所以,我觉得有责任更新我的回复。

首先,回答您提供的示例:

是的,这是可能的。

您可以通过更新 Child'sonClick解决这个问题this.props.onClick.bind(null, this)

var Child = React.createClass({
  render: function () {
    return <a onClick={this.props.onClick.bind(null, this)}>Click me</a>;
  }
});

Parent 中的事件处理程序然后可以像这样访问组件和事件:

  onClick: function (component, event) {
    // console.log(component, event);
  },

JSBin 快照


但问题本身具有误导性

Parent 已经知道 Child 的props.

这在提供的示例中并不清楚,因为实际上没有提供props。此示例代码可能会更好地支持所提出的问题:

var Child = React.createClass({
  render: function () {
    return <a onClick={this.props.onClick}> {this.props.text} </a>;
  }
});

var Parent = React.createClass({
  getInitialState: function () {
    return { text: "Click here" };
  },
  onClick: function (event) {
    // event.component.props ?why is this not available? 
  },
  render: function() {
    return <Child onClick={this.onClick} text={this.state.text} />;
  }
});

在这个例子中,你已经知道 Child 的 props 是什么变得更加清楚了。

JSBin 快照


如果真的是为了使用孩子的props……

如果真的是关于使用 Child 的props,你可以完全避免与 Child 的任何连接。

JSX 有一个扩展属性API,我经常在像 Child 这样的组件上使用。它需要所有的props并将它们应用到一个组件上。孩子看起来像这样:

var Child = React.createClass({
  render: function () {
    return <a {...this.props}> {this.props.text} </a>;
  }
});

允许您直接在父级中使用这些值:

var Parent = React.createClass({
  getInitialState: function () {
    return { text: "Click here" };
  },
  onClick: function (text) {
    alert(text);
  },
  render: function() {
    return <Child onClick={this.onClick.bind(null, this.state.text)} text={this.state.text} />;
  }
});

JSBin 快照


连接额外的子组件时不需要额外的配置

var Parent = React.createClass({
  getInitialState: function () {
    return {
      text: "Click here",
      text2: "No, Click here",
    };
  },
  onClick: function (text) {
    alert(text);
  },
  render: function() {
    return <div>
      <Child onClick={this.onClick.bind(null, this.state.text)} text={this.state.text} />
      <Child onClick={this.onClick.bind(null, this.state.text2)} text={this.state.text2} />
    </div>;
  }
});

JSBin 快照

但我怀疑这不是您的实际用例。所以让我们进一步挖掘......


一个健壮的实际例子

所提供示例的通用性质很难讨论。我创建了一个组件来演示上述问题的实际用途,以一种非常react的方式实现:

DTServiceCalculator 工作示例
DTServiceCalculator repo

该组件是一个简单的服务计算器。您向它提供服务列表(包括名称和价格),它会计算所选价格的总和。

孩子们幸福地无知

ServiceItem是本例中的子组件。它对外界的看法并不多。需要一些 props,其中之一是单击时要调用的函数。

<div onClick={this.props.handleClick.bind(this.props.index)} />

它只是handleClick使用提供的index[ source ]调用提供的回调

父母是孩子

DTServicesCalculator是父组件就是这个例子。这也是个孩子。我们看看吧。

DTServiceCalculator创建一个子组件列表,ServiceItem并为它们提供 props [ source ]。它是 的父组件,ServiceItem但它是传递列表的组件的子组件。它不拥有数据。所以它再次将组件的处理委托给它的父组件

<ServiceItem chosen={chosen} index={i} key={id} price={price} name={name} onSelect={this.props.handleServiceItem} />

handleServiceItem捕获从子项传递的索引,并将其提供给其父项 [来源]

handleServiceClick (index) {
  this.props.onSelect(index);
}

楼主什么都知道

“所有权”的概念是 React 中的一个重要概念。我建议在此处阅读更多相关信息

在我展示的示例中,我一直将事件的处理委托给组件树,直到我们到达拥有状态的组件。

当我们最终到达那里时,我们像这样处理状态选择/取消选择 [来源]:

handleSelect (index) {
  let services = […this.state.services];
  services[index].chosen = (services[index].chosen) ? false : true;
  this.setState({ services: services });
}


结论

尽量保持最外面的组件不透明。努力确保他们对父组件如何选择实现它们的偏好很少。

随时了解谁拥有您正在操纵的数据。在大多数情况下,您需要将树上的事件处理委托给拥有该状态的组件

旁白:Flux 模式是减少应用程序中此类必要连接的好方法。

感谢简洁的回答,确实帮助我更好地理解了 React!我有一个同样的问题。我正在将 Flux 用于发布/订阅。不是像您的示例那样将 Child 事件处理程序传递给 Parent,而是可以将其实现为“动作”并监听它。你认为这是一个很好的替代品和使用 Flux 吗?
2021-03-15 02:16:30
onClick={this.onClick.bind(null, this.state.text)}不需要绑定状态,因为父级有。所以只会onClick={this.onClick}工作。onClick = ()=>{const text = this.state.text;..}父母在哪里
2021-03-23 02:16:30
谢谢@Pathsofdesign!这将取决于。Flux 有这个控制器视图的概念。在这个例子中,Parent可能是这样一个控制器视图,而Child只是一个哑视图(组件)。只有控制器视图应该了解应用程序。在这种情况下,您仍然会将 Action from ParenttoChild作为 prop传递就 Action 本身而言,Flux 有一个强烈规定的模式,用于在 Action 之间进行交互以更新视图。facebook.github.io/flux/docs/...
2021-03-26 02:16:30

似乎有一个简单的答案。考虑一下:

var Child = React.createClass({
  render: function() {
    <a onClick={this.props.onClick.bind(null, this)}>Click me</a>
  }
});

var Parent = React.createClass({
  onClick: function(component, event) {
    component.props // #=> {Object...}
  },
  render: function() {
    <Child onClick={this.onClick} />
  }
});

关键是调用从父级传递bind(null, this)this.props.onClick事件。现在, onClick 函数接受参数component, AND event我认为这是世界上最好的。

更新:2015 年 9 月 1 日

这是一个坏主意:让子实现细节泄露给父从来不是一条好路。请参阅 Sebastien Lorber 的回答。

@TheSurrican 不,完全没问题。第一个参数绑定到this函数内(无论如何都不需要),第二个参数在调用 onClick 时作为第一个参数。
2021-03-18 02:16:30
不是以错误的方式争论(交换位置)吗?
2021-03-21 02:16:30

问题是如何将参数从子组件传递给父组件。这个例子很容易使用和测试:

//Child component
class Child extends React.Component {
    render() {
        var handleToUpdate  =   this.props.handleToUpdate;
        return (<div><button onClick={() => handleToUpdate('someVar')}>Push me</button></div>
        )
    }
}

//Parent component
class Parent extends React.Component {
    constructor(props) {
        super(props);
        var handleToUpdate  = this.handleToUpdate.bind(this);
    }

    handleToUpdate(someArg){
        alert('We pass argument from Child to Parent: \n' + someArg);
    }

    render() {
        var handleToUpdate  =   this.handleToUpdate;
        return (<div>
          <Child handleToUpdate = {handleToUpdate.bind(this)} />
        </div>)
    }
}

if(document.querySelector("#demo")){
    ReactDOM.render(
        <Parent />,
        document.querySelector("#demo")
    );
}

看看JSFIDDLE

就我而言,它忘记将回调函数传递给 click()
2021-04-05 02:16:30

基本上,您使用 props 向 Child 和 Parent 发送信息或从 Child 和 Parent 发送信息。

添加到所有精彩的答案,让我举一个简单的例子来解释在 React 中将值从子组件传递到父组件

应用程序.js

class App extends React.Component {
      constructor(){
            super();
            this.handleFilterUpdate = this.handleFilterUpdate.bind(this);
            this.state={name:'igi'}
      }
      handleFilterUpdate(filterValue) {
            this.setState({
                  name: filterValue
            });
      }
   render() {
      return (
        <div>
            <Header change={this.handleFilterUpdate} name={this.state.name} />
            <p>{this.state.name}</p>
        </div>
      );
   }
}

头文件.js

class Header extends React.Component {
      constructor(){
            super();
            this.state={
                  names: 'jessy'
            }
      }
      Change(event) {

      // this.props.change(this.state.names);
      this.props.change('jessy');
  }

   render() {
      return (
       <button onClick={this.Change.bind(this)}>click</button>

      );
   }
}

主程序

import React from 'react';
import ReactDOM from 'react-dom';

import App from './App.jsx';

ReactDOM.render(<App />, document.getElementById('app'));

就是这样,现在您可以将值从客户端传递到服务器。

看一下Header.js中的Change函数

Change(event) {
      // this.props.change(this.state.names);
      this.props.change('jessy');
  }

这就是您将值从客户端推送到服务器的方式

我可以知道这是你如何将值从客户端推送到道具到服务器,你说的这个语句是什么意思
2021-03-26 02:16:30