- What is prop drilling?
- What is lifting the state up?
- What are Context Provider and Context Consumer?
- If you don't pass a value to the provider does it take the default value?
Ans: Prop drilling is a way sending props to a child component which is many components below the parent component.
<ParentComponent/>
<ChildComponent/>
<SubChildComponent/>
If we need to pass props from ParentComponent
to SubChildComponent
, we would require to send the props though ChildComponent
also.
Prop drilling in React refers to the process of passing data (props) from a parent component to a deeply nested child component through intermediate components that do not necessarily need the data themselves. This can result in a situation where many components in the component tree are only serving as intermediaries to pass down props.
- Parent Component: The parent component holds the state or data that needs to be passed down.
- Intermediate Components: These components are between the parent and the final child component that needs the data. They receive the props and pass them down to their children.
- Child Component: The deeply nested component that actually uses the props.
import React, { useState } from "react";
// Parent Component
const App = () => {
const [data, setData] = useState("Hello from App");
return (
<div>
<IntermediateComponent data={data} />
</div>
);
};
// Intermediate Component
const IntermediateComponent = ({ data }) => {
return (
<div>
<AnotherIntermediateComponent data={data} />
</div>
);
};
// Another Intermediate Component
const AnotherIntermediateComponent = ({ data }) => {
return (
<div>
<ChildComponent data={data} />
</div>
);
};
// Child Component
const ChildComponent = ({ data }) => {
return <div>{data}</div>;
};
export default App;
In this example, the data
state from the App
component is passed down through IntermediateComponent
and AnotherIntermediateComponent
before reaching the ChildComponent
, which is where it is actually used.
- Increased Complexity: As the component tree grows, prop drilling can make the code harder to maintain and understand.
- Unnecessary Re-renders: Intermediate components that don't need the props will still receive them, potentially causing unnecessary re-renders.
- Tight Coupling: Components become tightly coupled, as changes in the structure of components can require changes in multiple places.
- Context API: React's Context API allows you to create a context object, which can be accessed by any component in the component tree without the need to pass props down manually.
import React, { useState, createContext, useContext } from "react";
const DataContext = createContext();
const App = () => {
const [data, setData] = useState("Hello from App");
return (
<DataContext.Provider value={data}>
<IntermediateComponent />
</DataContext.Provider>
);
};
const IntermediateComponent = () => {
return <AnotherIntermediateComponent />;
};
const AnotherIntermediateComponent = () => {
return <ChildComponent />;
};
const ChildComponent = () => {
const data = useContext(DataContext);
return <div>{data}</div>;
};
export default App;
- State Management Libraries: Libraries like Redux or MobX manage state globally, allowing components to access the state directly without passing props through intermediate components.
- Component Composition: Design components in a way that they don’t rely on deeply nested structures, reducing the need for prop drilling.
Ans: Lifting state up in React refers to the practice of moving state to a common ancestor component so that the state can be shared among multiple child components. This technique is useful when multiple components need to share or synchronize some state.
- State Sharing: When two or more sibling components need to share state, lifting the state up to their common parent allows them to access and modify the state.
- Single Source of Truth: Having a single source of truth for the state helps maintain data consistency and makes it easier to manage the state.
- Component Reusability: Components that don’t manage their own state are more reusable and easier to test.
- Identify the Common Ancestor: Determine the closest common ancestor component that can hold the shared state.
- Move State to Common Ancestor: Move the state and any related functions to this common ancestor.
- Pass State and Handlers Down: Pass the state and state updater functions as props to the child components that need them.
Suppose we have two sibling components, TemperatureInput
and BoilingVerdict
, that need to share the temperature state.
import React, { useState } from "react";
const TemperatureInput = ({ temperature, onTemperatureChange }) => {
const handleChange = (e) => {
onTemperatureChange(e.target.value);
};
return (
<fieldset>
<legend>Enter temperature in Celsius:</legend>
<input value={temperature} onChange={handleChange} />
</fieldset>
);
};
const BoilingVerdict = ({ celsius }) => {
if (celsius >= 100) {
return <p>The water would boil.</p>;
}
return <p>The water would not boil.</p>;
};
const Calculator = () => {
const [temperature, setTemperature] = useState("");
const handleTemperatureChange = (newTemperature) => {
setTemperature(newTemperature);
};
return (
<div>
<TemperatureInput
temperature={temperature}
onTemperatureChange={handleTemperatureChange}
/>
<BoilingVerdict celsius={parseFloat(temperature)} />
</div>
);
};
export default Calculator;
- Common Ancestor: The Calculator component is the common ancestor of TemperatureInput and BoilingVerdict.
- State in Common Ancestor: The temperature state is moved to the Calculator component.
- Passing Props: The temperature state and the handleTemperatureChange function are passed down to TemperatureInput as props.
- Shared State: Both TemperatureInput and BoilingVerdict now share the temperature state via the Calculator component.
- Consistency: Both components are synchronized and display consistent information.
- Separation of Concerns: Each component focuses on its specific functionality without managing the shared state.
- Easier Maintenance: Having a single source of truth for the state makes it easier to manage and debug the application.
Lifting state up is a fundamental pattern in React that helps in creating a predictable and maintainable data flow in your application.
Ans: In React, the Context API is a powerful feature that allows you to share values (such as state, functions, or any data) across the component tree without having to pass props down manually at every level. The Context.Provider
and Context.Consumer
are key parts of this API.
1. Context Provider
: Context Provider is used to provide access to a context between multiple components of the application. We can provide the access to the context or the data layer to the whole application to the and its subcomponents.
Example:
return (
<UserContext.Provider value={{ user: user, setUser: setUser }}>
<Head />
<Outlet />
<Footer />
</UserContext.Provider>
);
- In the code above , we are providing the access of UserContext to <Head /> <Outlet /> and <Footer /> component
2. Context Consumer:
is used to consume the context data , provided by react context. We can do this using useContext
hook for functional components and Context.Consumer
in clasees based components.
Example:
-
In Class based components, we can use the Context and use the consumer.
<UserContext.Consumer> {({ user }) => <h1>{user.name}</h1>} </UserContext.Consumer>
-
In Functional components we can use
useContext
hook to consume the context.import UserContext from "../utils/UserContext"; import { useContext } from "react"; const { user } = useContext(UserContext);
The Context.Provider
component is used to wrap a part of the component tree and make a value available to all descendant components. It acts as the source of the context values.
Create a Context
: Use React.createContext() to create a context object.Provide a Value
: Use the Provider component to wrap the part of the component tree where you want the context to be available and pass the value you want to share.Access the Value
: Descendant components can access the context value using the Context.Consumer component or the useContext hook.
import React, { createContext, useState, useContext } from "react";
// Create a Context
const MyContext = createContext();
const MyProvider = ({ children }) => {
const [value, setValue] = useState("Hello, World!");
return (
<MyContext.Provider value={{ value, setValue }}>
{children}
</MyContext.Provider>
);
};
const ChildComponent = () => {
const context = useContext(MyContext); // Using useContext hook
return (
<div>
<p>{context.value}</p>
<button onClick={() => context.setValue("New Value!")}>
Change Value
</button>
</div>
);
};
const App = () => {
return (
<MyProvider>
<ChildComponent />
</MyProvider>
);
};
export default App;
The Context.Consumer
component is used to access the context value in class components or if you prefer not to use hooks. It requires a function as its child, which receives the context value.
const AnotherChildComponent = () => {
return (
<MyContext.Consumer>
{(context) => (
<div>
<p>{context.value}</p>
<button onClick={() => context.setValue("Another New Value!")}>
Change Value
</button>
</div>
)}
</MyContext.Consumer>
);
};
- Context Provider (
Context.Provider
): Wraps a component tree and provides a context value to all descendant components. - Context Consumer (
Context.Consumer
): A component or hook (useContext
) that allows components to access the context value provided by the nearest Provider.
Using the Context API helps avoid prop drilling by allowing you to share values between components that are not directly connected in the component hierarchy.
Ans: If we do not override the values of context it takes the default values form the context , when we initialise the context.
Yes, if you do not pass a value to the Context.Provider
, it will take the default value defined when you created the context using React.createContext()
.
When you create a context using React.createContext(defaultValue)
, the defaultValue
is used if a component does not have a matching Provider
above it in the tree. This can be useful for providing a fallback value or ensuring that your components have a default state to work with.
Let's create a context with a default value and see what happens if we do not provide a value to the Provider
.
import React, { createContext, useContext } from "react";
// Create a Context with a default value
const MyContext = createContext("Default Value");
const DisplayValue = () => {
const value = useContext(MyContext);
return <div>{value}</div>;
};
const App = () => {
return (
<div>
<h1>Without Provider</h1>
<DisplayValue />
<h1>With Provider</h1>
<MyContext.Provider value="Provided Value">
<DisplayValue />
</MyContext.Provider>
</div>
);
};
export default App;
-
Creating the Context:
const MyContext = createContext("Default Value");
creates a context with a default value of"Default Value"
. -
Using the Context Without Provider: When
DisplayValue
is rendered outside of anyMyContext.Provider
, it uses the default value ("Default Value"
). -
Using the Context With Provider: When DisplayValue is rendered inside a
MyContext.Provider
with a value ("Provided Value"
), it uses the provided value instead of the default value.