React Lifecycle methods:
constructor()
getDerivedStateFromProps()
render()
componentDidMount()
A component is updated whenever there is a change in the component's state
or props
.
React has five built-in methods that gets called, in this order, when a component is updated:
getDerivedStateFromProps()
shouldComponentUpdate()
render()
getSnapshotBeforeUpdate()
componentDidUpdate()
In the getSnapshotBeforeUpdate()
method you have access to the props
and state
before the update, meaning that even after the update, you can check what the values were before the update.
If the getSnapshotBeforeUpdate()
method is present, you should also include the componentDidUpdate()
method, otherwise you will get an error.
the most reason for writing a method as a class property is when the method will be passed as a callback, and you need it to always be bound to the instance. React lifecycle methods will always be called as a method, so there's no reason to bind them (and you incur a tiny memory penalty when you do). Where this makes a difference is when you're passing a function to a component as a callback (e.g. onClick
or onChange
).
Take this example:
class BrokenFoo extends React.Component {
handleClick() {
alert(this.props.message);
}
render() {
return (
<button onClick={this.handleClick}>
Click me
</button>
)
}
}
The function represented by this.handleClick
is not automatically bound to the component instance, so when the method tries to read the value of this.props
it will throw a TypeError
because this
is not defined. Read this article if you're not familiar with this
; the problem described in section 4.2 "Pitfall: extracting methods improperly" is essentially what's happening when you pass around a method without making sure it's bound correctly.
Here's the class, rewritten with the handler as a class property:
class HappyFoo extends React.Component {
handleClick = () => {
alert(this.props.message);
}
render() {
return (
<button onClick={this.handleClick}>
Click me
</button>
)
}
}
Effectively, you can think of the handleClick
definition in the second example as placing this code into the component's constructor (which is just about exactly the way Babel does it):
this.handleClick = () => {
alert(this.props.message);
}
This achieves the same thing as calling bind
on the function (as described in the linked article) but does it a little differently. Because this function is defined in the constructor, the value of this
in this.props.message
is bound to the containing instance. What this means is that the function is now independent of the calling context; you can pass it around and it won't break.
The rule of thumb that I follow: by default, write methods as methods. This attaches the method to the prototype
and will usually behave the way you'd expect. However, if the method is ever written without parentheses (i.e. you're passing the value and not calling it), then you likely want to make it a class property.
Comments
Post a Comment