Debunking the Myth: React and Immutability
The title might be a bit provocative, but it captures a common misconception about React. Many blog posts and articles mention React in the same breath as immutability, unidirectional data flow, and pure or functional components. This often leads newcomers to believe that React inherently manages immutability and purity as seen in functional programming languages. In reality, this is not the case.
React is a JavaScript library designed to work seamlessly with JavaScript, a language that is not purely functional. Consequently, React is not limited to purely functional paradigms. While React encourages certain best practices, like immutability, it's the developer's responsibility to implement them correctly.
Let's explore some key concepts in React to clarify these misunderstandings, shall we?
The First Misconception: Component vs. PureComponent
React initially introduced class-based components, where developers extended the Component
class to create components. Here's a basic example borrowed from react.dev:
1class Greeting extends Component {2render() {3return <h1>Hello, {this.props.name}!</h1>;4}5}
Later, React introduced PureComponent
. The main difference, as described in react.dev is:
PureComponent is similar to Component but it skips re-renders for same props and state.
This suggests that PureComponent only re-renders when there are changes in props or state, making it deterministic. Here's an example:
1class Greeting extends PureComponent {2render() {3return <h1>Hello, {this.props.name}!</h1>;4}5}
Greeting
component is pure because it does rerender only when props change, and it is deterministic because it will always render the same output for the same input - it's a function in the mathematical sense: can be memoized and paralelized.
Now, let's have a look at the definition of a pure component
...it skips re-renders for same props and state.
and state?
This introduces a subtle problem. Pure functions, by definition, cannot have internal state because state makes outputs non-deterministic. Therefore, PureComponent
is not truly a pure function, as it can still hold state, challenging the idea that it behaves like a pure function.
The Evolution: /Function(al)? components/
With the advent of Function Components, React developers gained a more efficient and streamlined way to build components. Function Components, combined with Hooks, offered a more powerful and flexible approach to handling state and side effects compared to class-based components.
However, somewhere along the line, Function Components began to be referred to as functional components in many articles. This led to the mistaken belief that React had fully embraced immutability and functional programming principles.
This misconception was further compounded by React's documentation, which outlined rules such as:
Components must be idempotent - React components are assumed to always return the same output with respect to their inputs - props, state, and context.
Here, React assumes that state and context are inputs, managed by Hooks like useState
and useContext
. I believe the reason is because their state is managed internally by React and not by an external entity or local variable. But what happens when we introduce side effects into these supposedly "pure" components?
The Purity Dilemma
Consider a custom hook usePersistence that writes to a synchronous persistence layer, such as local storage:
1function usePersistence<T>(initialValue: T): [T, (value: T) => void] {2// Implementation that writes to local storage3}
This hook introduces side effects by altering an external system. Despite React's guidelines, state and context managed through Hooks can still lead to non-idempotent behavior.
If you're not convinced, let's take the following example from react.dev:
1import { useState } from 'react';23export default function Counter() {4const [count, setCount] = useState(0);56function handleClick() {7setCount(count + 1);8}910return (11<button onClick={handleClick}>12You pressed me {count} times13</button>14);15}
This function component is not guaranteed to produce the same output every time it renders. The output depends on user interactions (e.g., clicking the button), making it inherently non-pure and non-idempotent.
From the perspective of the computation, this function has side-effects because it holds state and alters an external system.
Are you now convinced?
A New Paradigm: Server components
React 18 introduced Server Components, a significant shift from traditional SSR (Server-Side Rendering). Unlike SSR, where the server sends HTML along with JavaScript that rehydrates and re-renders on the client side, Server Components only send HTML to the client. This approach skips the client-side rehydration, reducing the amount of JavaScript sent to the client and improving performance.
Key Principles of Server Components
Server Components come with a set of rules that differentiate them from traditional components:
- No State or Side Effects: Server Components cannot hold state or execute side effects on the frontend. They can, however, interact with server-side functions like database queries or file system operations.
- No stateful Hooks: Server Components cannot use
useState
,useContext
, or any other Hook that derives from them. - Interoperability: Server Components can call server-side APIs and pass data to Client Components, but they must adhere to the separation of client and server logic.
Here's a simple example illustrating Server and Client Components:
1'use client'23function Clock() {4// empty initial state5const [time, setTime] = useState()67useEffect(() => {8const id = setInterval(() => setTime(new Date()), 1000)9return () => clearInterval(id)10}, [])1112// SSR will render nothing13return time ? <time>{time.toLocaleString()}</time> : null14}
1function App() {2// hypothetical call to a database3// assumes user is logged in4const user = db.getCurrentUser()5return (6<html>7<body>8Hello {user.name}, current time is: <Clock />9</body>10</html>11)12}
The Clock component is a Client Component with side effects, while App is a Server Component that renders static HTML based on server-side data.
Server components is absolutely new in React and also ressembles the way that we actually did frontend a decade ago. Also, this mix of Server and Client Components is also a response to a trend that were being filled in other frameworks called islands.
A Framework for Purity
Something I noticed is that Server Components offer React developers the ability to create truly pure, idempotent components by managing the separation of concerns between the server and client. By using Server Components for rendering static content and limiting state management to Client Components, developers can build more predictable, bug-free and maintainable applications.
It really depends on how to manage your set of components.
Taking the silly examples above, here's an example of how you might refactor your codebase:
1function Greet({ name }) {2return <>Hello {name}</>3}
1function ShowCounter({ count }) {2return <>You pressed me {count} times</>;3}
1'use client'23function Counter() {4const [count, setCount] = useState(0);56return (7<button onClick={() => setCount(count + 1)}>8<ShowCounter count={count} />9</button>10);11}
1function App() {2// server-side call to a database3// assumes user is logged in4const user = db.getCurrentUser()5return (6<html>7<body>8<Greet {...user} />, current time is: <Clock />9<Counter />10</body>11</html>12)13}
In this setup, Greet
and ShowCounter
are pure components that can be used anywhere. Counter
manages state and interacts with components that handle state mutations. The App
component handles all server-side effects, ensuring a clean separation between pure and impure logic.
Conclusion
By embracing Server Components, React developers can better manage state and UI updates, minimizing bugs and enhancing performance. I truly believe that this division of components into pure functions and stateful logic helps create a more maintainable and predictable codebase.