From 80a7f07a7b0ccdd84c1d21cbd71be04be772ab07 Mon Sep 17 00:00:00 2001 From: jim Date: Wed, 6 Jan 2016 05:09:56 -0800 Subject: [PATCH] Blog post explains to verify prop mutations in componentWillReceiveProps --- ...-A-implies-B-does-not-imply-B-implies-A.md | 60 +++++++++++++++++++ docs/ref-03-component-specs.md | 2 + 2 files changed, 62 insertions(+) create mode 100644 _posts/2016-01-08-A-implies-B-does-not-imply-B-implies-A.md diff --git a/_posts/2016-01-08-A-implies-B-does-not-imply-B-implies-A.md b/_posts/2016-01-08-A-implies-B-does-not-imply-B-implies-A.md new file mode 100644 index 00000000..29c8aedb --- /dev/null +++ b/_posts/2016-01-08-A-implies-B-does-not-imply-B-implies-A.md @@ -0,0 +1,60 @@ +--- +title: "(A => B) !=> (B => A)" +author: jimfb +--- + +The documentation for `componentWillReceiveProps` states that `componentWillReceiveProps` will be invoked when the props change as the result of a rerender. Some people assume this means "if `componentWillReceiveProps` is called, then the props must have changed", but that conclusion is logically incorrect. + +The guiding principle is one of my favorites from formal logic/mathematics: + > A implies B does not imply B implies A + +Example: "If I eat moldy food, then I will get sick" does not imply "if I am sick, then I must have eaten moldy food". There are many other reasons I could be feeling sick. For instance, maybe the flu is circulating around the office. Similarly, there are many reasons that `componentWillReceiveProps` might get called, even if the props didn’t change. + +If you don’t believe me, call `ReactDOM.render()` three times with the exact same props, and try to predict the number of times `componentWillReceiveProps` will get called: + + +```js +class Component extends React.Component { + componentWillReceiveProps(nextProps) { + console.log('componentWillReceiveProps', nextProps.data.bar); + } + render() { + return
Bar {this.props.data.bar}!
; + } +} + +var container = document.getElementById('container'); + +var mydata = {bar: 'drinks'}; +ReactDOM.render(, container); +ReactDOM.render(, container); +ReactDOM.render(, container); +``` + + +In this case, the answer is "2". React calls `componentWillReceiveProps` twice (once for each of the two updates). Both times, the value of "drinks" is printed (ie. the props didn’t change). + +To understand why, we need to think about what *could* have happened. The data *could* have changed between the initial render and the two subsequent updates, if the code had performed a mutation like this: + +```js +var mydata = {bar: 'drinks'}; +ReactDOM.render(, container); +mydata.bar = 'food' +ReactDOM.render(, container); +mydata.bar = 'noise' +ReactDOM.render(, container); +``` + +React has no way of knowing that the data didn’t change. Therefore, React needs to call `componentWillReceiveProps`, because the component needs to be notified of the new props (even if the new props happen to be the same as the old props). + +You might think that React could just use smarter checks for equality, but there are some issues with this idea: + + * The old `mydata` and the new `mydata` are actually the same physical object (only the object’s internal value changed). Since the references are triple-equals-equal, doing an equality check doesn’t tell us if the value has changed. The only possible solution would be to have created a deep copy of the data, and then later do a deep comparison - but this can be prohibitively expensive for large data structures (especially ones with cycles). + * The `mydata` object might contain references to functions which have captured variables within closures. There is no way for React to peek into these closures, and thus no way for React to copy them and/or verify equality. + * The `mydata` object might contain references to objects which are re-instantiated during the parent's render (ie. not triple-equals-equal) but are conceptually equal (ie. same keys and same values). A deep-compare (expensive) could detect this, except that functions present a problem again because there is no reliable way to compare two functions to see if they are semantically equivalent. + +Given the language constraints, it is sometimes impossible for us to achieve meaningful equality semantics. In such cases, React will call `componentWillReceiveProps` (even though the props might not have changed) so the component has an opportunity to examine the new props and act accordingly. + +As a result, your implementation of `componentWillReceiveProps` MUST NOT assume that your props have changed. If you want an operation (such as a network request) to occur only when props have changed, your `componentWillReceiveProps` code needs to check to see if the props actually changed. + + diff --git a/docs/ref-03-component-specs.md b/docs/ref-03-component-specs.md index 2855caa7..6e00ad52 100644 --- a/docs/ref-03-component-specs.md +++ b/docs/ref-03-component-specs.md @@ -148,6 +148,8 @@ componentWillReceiveProps: function(nextProps) { > Note: > +> One common mistake is for code executed during this lifecycle method to assume that props have changed. To understand why this is invalid, read [A implies B does not imply B implies A](/react/blog/2016/01/08/A-implies-B-does-not-imply-B-implies-A.html) +> > There is no analogous method `componentWillReceiveState`. An incoming prop transition may cause a state change, but the opposite is not true. If you need to perform operations in response to a state change, use `componentWillUpdate`.