Both Jotai and Recoil are state management libraries for React that use an atomic approach, meaning they break down state into small, independent pieces. They help developers manage complex application state without the boilerplate of Redux. These libraries are especially good for when you need to share data between different components without prop drilling.
Both jotai and recoil are state management libraries for React. They provide a way to manage global state by creating atoms (jotai) or atoms and selectors (recoil). Recoil has more features out of the box, such as caching and optimistic updates, while jotai is more lightweight and flexible.
Both jotai and recoil have excellent TypeScript support, with jotai having a slight edge in terms of type inference.
Both libraries are compatible with modern browsers, including Chrome, Firefox, and Edge.
jotai has no dependencies, while recoil depends on react and react-dom.
jotai is generally faster and more lightweight than recoil due to its smaller bundle size and simpler architecture.
Both libraries are compatible with React, but recoil is more tightly integrated with the React ecosystem.
Recoil has a more active community, with more contributors and issues on GitHub.
Both libraries have excellent documentation, but recoil's documentation is more comprehensive and up-to-date.
Recoil is maintained by Facebook, while jotai is maintained by a single developer.
1import { atom, useAtom } from 'jotai';
2const countAtom = atom(0);
3function Counter() {
4 const [count, setCount] = useAtom(countAtom);
5 return (
6 <div>
7 <p>Count: {count}</p>
8 <button onClick={() => setCount(count + 1)}>Increment</button>
9 </div>
10 );
11}
This example creates a simple counter using jotai's atom and useAtom hooks.
1import { atom, selector, useRecoilState } from 'recoil';
2const countAtom = atom({
3 key: 'count',
4 default: 0,
5});
6function Counter() {
7 const [count, setCount] = useRecoilState(countAtom);
8 return (
9 <div>
10 <p>Count: {count}</p>
11 <button onClick={() => setCount(count + 1)}>Increment</button>
12 </div>
13 );
14}
This example creates a simple counter using recoil's atom and useRecoilState hooks.
jotai is a better choice for smaller applications or those that require a more lightweight state management solution, while recoil is a better choice for larger applications or those that require more advanced features.
A small, fast state management solution that uses hooks. It's super simple to learn and doesn't require much setup code like other state managers.
Like Jotai and Recoil, it's modern and hook-based. It's becoming very popular because it's easier to learn than Redux and has a tiny bundle size. Perfect for small to medium React apps.
State ManagementA proxy-based state management tool that makes your React state work like regular JavaScript objects. You can update state directly without special actions or reducers.
It's made by the same team as Jotai and shares similar principles of simplicity. Great for developers who want their state management to feel more natural and less framework-specific.
State ManagementA tiny state manager that works with any framework, not just React. It's super lightweight and focuses on simplicity and speed.
While not as React-specific as Jotai or Recoil, it offers atomic state management in a similar way. It's perfect for those who want something extremely lightweight or might work with multiple frameworks.
State ManagementA battle-tested state management library that makes state management as simple as working with plain JavaScript objects. It uses a reactive programming approach.
While it's older than Jotai and Recoil, it offers similar benefits of simple state management with less boilerplate than Redux. Good for larger applications that need more features.
State Managementvisit jotai.org or npm i jotai
Jotai scales from a simple useState replacement to an enterprise TypeScript application.
An atom represents a piece of state. All you need is to specify an initial value, which can be primitive values like strings and numbers, objects, and arrays. You can create as many primitive atoms as you want.
import { atom } from 'jotai' const countAtom = atom(0) const countryAtom = atom('Japan') const citiesAtom = atom(['Tokyo', 'Kyoto', 'Osaka']) const mangaAtom = atom({ 'Dragon Ball': 1984, 'One Piece': 1997, Naruto: 1999 })
It can be used like React.useState
:
import { useAtom } from 'jotai' function Counter() { const [count, setCount] = useAtom(countAtom) return ( <h1> {count} <button onClick={() => setCount((c) => c + 1)}>one up</button> ...
A new read-only atom can be created from existing atoms by passing a read
function as the first argument. get
allows you to fetch the contextual value
of any atom.
const doubledCountAtom = atom((get) => get(countAtom) * 2) function DoubleCounter() { const [doubledCount] = useAtom(doubledCountAtom) return <h2>{doubledCount}</h2> }
You can combine multiple atoms to create a derived atom.
const count1 = atom(1) const count2 = atom(2) const count3 = atom(3) const sum = atom((get) => get(count1) + get(count2) + get(count3))
Or if you like fp patterns ...
const atoms = [count1, count2, count3, ...otherAtoms] const sum = atom((get) => atoms.map(get).reduce((acc, count) => acc + count))
You can make the read function an async function too.
const urlAtom = atom('https://json.host.com') const fetchUrlAtom = atom(async (get) => { const response = await fetch(get(urlAtom)) return await response.json() }) function Status() { // Re-renders the component after urlAtom is changed and the async function above concludes const [json] = useAtom(fetchUrlAtom) ...
Specify a write function at the second argument. get
will return the current
value of an atom. set
will update the value of an atom.
const decrementCountAtom = atom( (get) => get(countAtom), (get, set, _arg) => set(countAtom, get(countAtom) - 1) ) function Counter() { const [count, decrement] = useAtom(decrementCountAtom) return ( <h1> {count} <button onClick={decrement}>Decrease</button> ...
Just do not define a read function.
const multiplyCountAtom = atom(null, (get, set, by) => set(countAtom, get(countAtom) * by), ) function Controls() { const [, multiply] = useAtom(multiplyCountAtom) return <button onClick={() => multiply(3)}>triple</button> }
Just make the write function an async function and call set
when you're ready.
const fetchCountAtom = atom( (get) => get(countAtom), async (_get, set, url) => { const response = await fetch(url) set(countAtom, (await response.json()).count) } ) function Controls() { const [count, compute] = useAtom(fetchCountAtom) return ( <button onClick={() => compute('http://count.host.com')}>compute</button> ...
Jotai's fluid interface is no accident — atoms are monads, just like promises! Monads are an established pattern for modular, pure, robust and understandable code which is optimized for change. Read more about Jotai and monads.
A lightweight local-first graphic-centric productivity tool to build your second brain. Supporting Excalidraw/Tldraw whiteboard and notion-like note. 一款以图形为中心、轻量级、本地优先的用于构建第二大脑的效率工具。支持 Excalidraw、Tldraw 白板和类 Notion 笔记。
A simple calendar with nice design for remarking a date
Invoicing for freelancers & small businesses
SVG based drawing tool and react-native component
A fully featured editor with drag and drop interface to easily build READMEs