React.PureComponent - different ways of writing event handlers

There are many ways to define event handlers in React. Are there any differences between them?

TLDR; You should probably avoid the inline arrow functions and handlers defined in the render() method.

Update 2020

For React Hooks there is no way to use class properties or class methods. React Hooks approach favours readability and ease of implementation over raw performance. One thing which is worth remembering is to wrap callbacks with useCallback, but please to this only when callback is passed to a memoized/Pure Components.
There is a very good article regarding this topic from Kent C. Dodds.

Overview

Writing React handlers is easy, just take care of "this" and you're good to go.

<button onClick={() => console.log("it works", this)}>Click me</button>

Are you sure?

Example

A brief example of the various handler defining methods and their impact on the re-rendering of React.PureComponent based elements.


NOTE: You may open console in above example to see a value of this for each handler.

Description

In the above example, you can see five ways to write click handlers.

Handler defined outside of a component

Easy to write, but you won't be able to access this from within the handler, that's why most of the time it is not so useful.

const handleClickOutsideClass = () => {
  console.log("click", this);
};

Handler as a class method

This is probably the most "classic" way of handling events in React components.

handleClickMethodBound() {
  console.log("click", this);
}

In most cases, you should remember to bind handler methods in the class constructor or remember that this will be undefined inside the handler. Here is a binding implementation example:

constructor(props) {
  super(props);
  this.handleClickMethodBound = this.handleClickMethodBound.bind(this);
}

While it might be important for your code to bind the handler method, it's also worth knowing that binding will slow down your code, a little bit, but still.

Handler as a class property

The most elegant way to define event handler is through the Babel class properties.

handleClickClassProp = () => {
  console.log("click", this);
};

In this method you can avoid binding inside the class constructor and still keep expected this assignment. Of course you will have to use @babel/plugin-proposal-class-properties in order to make this work. One more drawback of this method is a Babel transpilation result. Class properties are transpiled to arrow functions in the class constructor. It means that you won't be able to access them from outside the class e.g. to mock/spy them in tests, or use with super in subclasses.

Handler inside render() method

In this method, a new handler function will be created on each re-render, which is probably not the desired behavior.

  render() {
    const handleClickInRender = () => {
      console.log("click", this);
    };
    // ... Remaining implementation

Handler as an inline arrow function

Arrow functions may quickly solve this related issues. This method has similar disadvantages to the handlers defined inside render() method.

return <button onClick={() => console.log("click", this)}>Click me</button>;

You can use the inline arrow functions to avoid binding class methods in the class constructor, but drawbacks are still the same. On each render, a new handler method will be created.

Summary

As you can see it is easy to create handler function on each render, which is not necessarily something you want to do. It seems that class methods with optional binding (if necessary) are most performant solution but on the other hand readability is in most cases should be the key factor. I would recommend class properties if only you can apply required Babel plugins to your codebase.

Sources