What is React? Challenging Your Knowledge by Going Beyond...
Challenging What You Thought You Knew by Going Beyond the Surface!
π In life, having answers to the what, why, and how of your goals greatly increases your chances of success. π
π― Let's consider the goal of becoming a front-end developer. It's crucial to know what frontend development entails, why you want to pursue it, and how you plan to achieve it. π
π€ However, when it comes to learning new technologies, we often overlook these important aspects. But don't worry, in this blog, we won't skip them. π
π Instead, we'll delve into what React is, why it came into existence, its purpose, and how it fulfills that purpose through core principles. We'll also explore its current limitations. π
π‘ But don't worry, this won't be just another definition-heavy blog. π Instead, we'll understand React through real-world analogies and explore optimal and suboptimal examples. We'll even tackle interview-related questions and provide hands-on exercises to reinforce your knowledge. πͺ
β¨οΈ By having answers to these questions, you'll approach coding with a clear mindset, avoiding the pitfalls of crashing or encountering unexpected behaviors. π§
Don't forget to share your own "why" in the comment section!
Introduction
π Welcome to my blog series! π
I'm on a mission to provide a comprehensive and practical understanding of React. π
By the end of the Mastering React series, you'll gain the knowledge and confidence to excel in React and overcome any obstacles. πͺ
Stay updated by following me along with subscribing to the newsletter so you Don't miss out on another blog!
Let's dive into React together and embark on an exciting journey! π
What is React? π€
React is a powerful JavaScript library used for building user interfaces. π It offers a declarative and efficient approach to creating interactive UI components.
Imagine React as a collection of building blocks𧱠that you can use to construct a house. Each block represents a React component, and when you combine these components, you can build a fully functional and interactive UI. π
As you delve deeper into React, you'll learn how to optimize this construction process for efficient and effective UI development. πͺ
Overview of React Purpose (Why) β
React was developed to overcome several limitations and challenges faced by previous web development technologies. These limitations include:
Complex HTML manipulation: Traditional frameworks required intricate manipulation of HTML, leading to complex and convoluted code.
Inefficient rendering: Dynamic UI updates were cumbersome, resulting in poor rendering performance.
Lack of reusability: Previous approaches lacked modularity, making it difficult to reuse code effectively.
Inconsistent data management: Handling application state was challenging, often leading to data inconsistencies.
Imperative code complexity: Imperative programming approaches resulted in bugs and difficulties in maintenance.
Weak support for single-page applications (SPAs): Existing technologies struggled with the efficient development of SPAs.
React addresses these limitations by introducing a component-based architecture, a virtual DOM for efficient updates, a one-way data flow for predictable state management, and a declarative programming approach. π
React Core Principles (How) ποΈ
Let's imagine you're constructing a house. React serves as the construction framework, helping you build different components of the house, such as walls, doors, and windows. π
Component-Based Architecture: In React, you break down your house into smaller components, just like Lego blocks. Each component represents a specific part of the house, such as a Wall component, a Door component, or a Window component. These components can be reused in different houses, similar to how Lego blocks can build various structures.
Virtual DOM and Reconciliation Process: React's Virtual DOM acts as a blueprint for your house. Instead of making changes directly to the physical house, React updates the blueprint first. It then intelligently determines which parts of the house need modifications and efficiently applies those changes. This process, known as reconciliation, ensures only necessary updates are made, resulting in faster and smoother construction.
Unidirectional Data Flow: Imagine data flow in React as instructions from an architect to construction workers. The architect (parent component) provides instructions to the workers (child components) on how to build each component. The workers follow these instructions and construct their respective components accordingly. This one-way flow of instructions simplifies the construction process and maintains consistency.
Declarative Approach: React's declarative approach is like to having an interior designer decorate your house. Rather than specifying every detail of how to decorate each room, you describe the desired outcome or ambiance. The interior designer (React) takes care of the underlying logic and applies the decorations accordingly. This allows you to focus on the overall vision without getting caught up in the specifics.
Immutability: In React, immutability is like having an unchanging blueprint. When you need to make changes, you create a new blueprint with the desired modifications. Similarly, in React, you create new objects or arrays when updating data instead of directly modifying existing ones. This ensures the state remains consistent and facilitates tracking changes.
Throughout this blog series, we'll illustrate each core principle using simple code examples. Don't worry if you're new to React or have never written a line of code before. The code itself will be self-explanatory, and you can try to understand it. Later on, we'll provide explanations to enhance your understanding. π
React Component-Based Architecture π
Let's imagine you're constructing a car using a component-based approach in React. π
Understanding the concept of components in React: In React, components serve as the building blocks of your car. Each component represents a specific part of the car, such as the engine, wheels, seats, and dashboard. Every component is responsible for its own functionality and appearance, making the car a cohesive and organized structure.
Benefits of a component-based approach: Embracing the component-based approach offers several advantages in building your car:
Reusability: Like components in React, certain parts of the car, such as the engine or wheels, can be reused across different car models. This eliminates duplication and saves time and effort in the manufacturing process.
Modularity: Components in React are independent and self-contained, allowing for easier maintenance and updates. Similarly, in the car manufacturing process, each component can be developed and tested separately, facilitating troubleshooting and continuous improvement.
Encapsulation: React components encapsulate their own logic and data, preventing interference with other components. Similarly, in the car, each component has its own internal mechanisms and functions, ensuring their functionalities work independently. For instance, the engine component handles power and performance without affecting other components like the seats or dashboard.
Collaboration: React enables multiple developers to work simultaneously on different components, fostering faster development and collaboration. Similarly, in the car manufacturing process, specialized teams can work on specific components, such as the electrical system or suspension, ensuring expertise and efficiency in each area.
As you proceed to create your next component, keep these benefits in mind, rather than merely fulfilling the component's immediate task. By doing so, you can harness the power of component-based architecture effectively.
It may seem like a lot of theory, but don't worry! We'll soon dive into practical examples to solidify your understanding. π οΈ
Here's the simplest code example that demonstrates the benefits of React component-based architecture by creating a simple car dashboard component:
Bad Code Example - Not Leveraging Benefits: π
// CarDashboard.js
import React from 'react';
function CarDashboard() {
const speed = 80;
const fuelLevel = 50;
return (
<div>
<h2>Car Dashboard</h2>
<p>Speed: {speed}</p>
<p>Fuel Level: {fuelLevel}</p>
</div>
);
}
export default CarDashboard;
Explanation:
In the above "Bad" code example, we do not leverage the benefits of React's component-based architecture. Instead, we directly render the speed and fuel level as plain text within the CarDashboard
component. This approach lacks reusability, modularity, and encapsulation, making it harder to maintain and extend the codebase.
Example of Optimal Code - Leveraging Benefits: π
// CarDashboard.js
import React from 'react';
import Speedometer from './Speedometer';
import FuelGauge from './FuelGauge';
function CarDashboard() {
return (
<div>
<h2>Car Dashboard</h2>
<Speedometer speed={80} />
<FuelGauge fuelLevel={50} />
</div>
);
}
export default CarDashboard;
In the above code, we have a CarDashboard
component that leverages the benefits of React's component-based architecture. It composes of smaller, reusable components (Speedometer
and FuelGauge
) to build a complete car dashboard UI. Each component focuses on a specific functionality (speed and fuel level) and encapsulates its own state and rendering logic.
Do answer this question in your way to reinforce your learning.
Interview Questions:
What advantages does React's component-based architecture offer?
How does the component-based approach in React promote code reusability?
Can you explain the concept of encapsulation in React components?
In what ways does React's component-based architecture facilitate collaboration among developers?
Hands-on Exercise:
π§ Let's reinforce your learning with a hands-on exercise. Follow these steps:
ποΈ Create a new React project using Create React App or your preferred method.
π οΈ Implement a
Speedometer
component that receives the current speed as a prop and displays it on the UI.π οΈ Implement a
FuelGauge
component that receives the fuel level as a prop and displays it on the UI.π Use the
Speedometer
andFuelGauge
components within theCarDashboard
component.π₯οΈ Render the
CarDashboard
component in your application and observe the displayed speed and fuel level.
π‘ After completing the exercise, don't forget to share your progress by tagging me with #ReactWithHarshal on socials. Your engagement is valuable in assessing the impact of this content and ensuring its effectiveness in supporting your learning journey. Additionally, feel free to reach out if you have any questions or need mentoring, project review, or consultation assistance. Keep up the great work, and happy coding!
Virtual DOM and Reconciliation Process π
Imagine you're a librarianπ§βπ« managing a library with a large collection of books using a virtual catalog and an efficient updating process.
Explanation of Virtual DOM and its role in React: React's Virtual DOM can be likened to a virtual catalog for your library. It serves as a lightweight representation of the actual DOM (Document Object Model) - the comprehensive structure of your library, consisting of countless books. The Virtual DOM enables React to keep track of changes made to the library in an efficient manner.
Just as a virtual catalog allows you to perform operations like searching for specific books or updating the catalog without physically handling each book, the Virtual DOM acts as an intermediary between your application's logic and the real DOM. React can interact with the Virtual DOM, perform necessary operations, and optimize the updating process without directly manipulating the real DOM.
How React efficiently updates the DOM through reconciliation: Let's consider the scenario of adding a new bookπ to your library. React employs a highly efficient updating process known as reconciliation to handle such changes.
When a change occurs, React compares the virtual catalog (Virtual DOM) with the physical catalog (real DOM) to identify differences or updates that need to be made. However, instead of meticulously inspecting every bookshelf and book, React utilizes its reconciliation algorithm to optimize the process.
By determining the minimum set of changes required based on the disparities between the Virtual DOM and the real DOM, React focuses solely on the relevant bookshelfπ where the new book is added. This targeted approach allows React to update the catalog efficiently, avoiding unnecessary modifications to unchanged bookshelves or books.
This reconciliation process saves timeβ±οΈ and resources, ensuring a more streamlined and efficient updating mechanism.
By drawing parallels between React's Virtual DOM and reconciliation process and the management of a library with a virtual catalog, we gain a clearer understanding of how React optimizes the updating of the real DOM through selective and efficient changes.
Thanks! for sticking to reading this far If you found this blog helpful, please show your support by liking and commenting. Your encouragement motivates me to write more blogs and helps spread the word to fellow developers.
To stay updated and follow this blog series, don't forget to follow me and subscribe to the newsletter. Stay tuned so you don't miss another blog!
Now, let's delve into a code example that demonstrates the concept of the Virtual DOM and the reconciliation process in React.
Example of Bad Code: π
import React, { useState } from 'react';
// Library component
const Library = () => {
const [books, setBooks] = useState([
{ id: 1, title: 'Book 1' },
{ id: 2, title: 'Book 2' },
{ id: 3, title: 'Book 3' },
]);
// Function to add a new book
const addBook = () => {
const newBook = `Book ${books.length + 1}`;
books.push(newBook); // BAD: Directly modifying the original array
setBooks(books); // BAD: Setting the same array reference
};
return (
<div>
<h2>Library</h2>
<ul>
{books.map((book) => (
<li key={"bookid"}>{book.title}</li>
))}
</ul>
<button onClick={addBook}>Add Book</button>
</div>
);
};
export default Library;
Explanation:
This code snippet showcases a suboptimal implementation of the bookshelf component, which can lead to issues and unexpected behavior. Let's examine the problems and discuss their implications:
Mutation of the original books array: The code directly modifies the original
books
array using thepush
method. This violates React principle of immutability, which recommends creating new objects or arrays instead of modifying existing ones. By mutating the array, the component loses track of changes and may not trigger proper re-rendering, resulting in an incorrect UI representation.Ineffective state update: The
setBooks(books)
call sets the state with the same array reference. Since React relies on shallow equality checks to determine state updates, it may not recognize the change and skip the re-rendering process. Consequently, the UI fails to reflect the addition of the new book accurately. This issue is common among beginner developers who may overlook the importance of immutability and state updates.Incorrect usage of the
key
prop: Within the Library component, there is a list of books rendered. However, thekey
prop, which is crucial for React reconciliation process, is not correctly assigned. Properly assigning unique keys to list items enables React to efficiently identify and update changes within the list. Neglecting the key assignment can lead to performance problems and difficulties in accurately tracking changes.
Example of Optimal Code: π
import React, { useState } from 'react';
// Bookshelf component
const Bookshelf = ({ books }) => {
return (
<ul>
{books.map((book) => (
<li key={book.id}>{book.title}</li>
))}
</ul>
);
};
// Library component
const Library = () => {
const [books, setBooks] = useState([
{ id: 1, title: 'Book 1' },
{ id: 2, title: 'Book 2' },
{ id: 3, title: 'Book 3' },
]);
const addBook = () => {
const newBook = { id: 4, title: 'Book 4' };
//Update data instead of directly modifying the existing ones
setBooks([...books, newBook]);
};
return (
<div>
<h2>Library</h2>
<Bookshelf books={books} />
<button onClick={addBook}>Add Book</button>
</div>
);
};
export default Library;
Explanation:
In this example, we have a Library component that manages a list of books. The Library component renders a Bookshelf component, which represents a collection of books. Whenever the "Add Book" button is clicked, a new book is added to the list of books using the addBook
function.
React's Virtual DOM and reconciliation process come into play when a book is added. React efficiently updates the DOM by comparing the previous Virtual DOM with the updated Virtual DOM. It identifies the specific changes (in this case, the addition of a new book) and only updates the necessary parts of the DOM, in this case, the <ul>
element within the Bookshelf component.
Interview Questions: β
π€ What is the purpose of the Virtual DOM in React and why is it used?
π€ Can you explain the reconciliation process in React and how it improves DOM updates?
π€ How does React determine the minimal set of changes needed during reconciliation?
π€ What are the advantages of using the Virtual DOM and reconciliation in terms of performance and user experience?
π€ How does React efficiently handle additions or removals of items in a list?
Hands-on Exercise:
π§ Let's dive into some hands-on practice to reinforce your understanding of the Virtual DOM and reconciliation in React:
ποΈ Set up a new React project using Create React App or your preferred method.
π Replace the default content of the App component with the provided code snippet for the Library component.
βΆοΈ Run the React application and observe how the initial list of books is rendered.
β Click the "Add Book" button and witness how React efficiently updates the DOM by adding the new book to the bookshelf without re-rendering the entire component.
By actively engaging in the code examples and answering the interview questions, you'll solidify your comprehension of the Virtual DOM, the reconciliation process, and their vital role in React development. The hands-on exercise enables you to witness these concepts in action, enhancing your learning experience.
Advantages of Using React π
If you've made it this far, you can figure out the advantages of using React. I will encourage you to take a moment and Feel free to share it in the comments section as well.
Considering the length of the blog so far I will keep this section shorter and concise and will write another blog covering in-depth react limitations with practical examples, how to tackle those limitations, interview-related questions, and hands-on exercises.
Understanding React Limitationsπ€·ββοΈ
React, while powerful, has certain limitations and trade-offs to consider:
Learning Curve: React has a learning curve, especially for beginners. It requires understanding concepts like components, props, state, and JSX syntax.
Complexity for Small Projects: React's complexity may be unnecessary for small, simple projects.
Performance Impact: Large-scale applications or complex component hierarchies in React may require optimization techniques to maintain performance.
Mobile Development: React Native is used for mobile app development, but highly performance-critical or platform-specific apps may benefit from native development or specialized frameworks.
SEO Limitations: JavaScript-rendered content in React apps can pose challenges for search engine optimization (SEO). Techniques like server-side rendering (SSR) or pre-rendering can mitigate this.
Tooling and Ecosystem: Staying up-to-date with React's ecosystem and tools can be overwhelming due to the rapidly evolving JavaScript landscape.
Summary β¨
π In this blog, we covered the what, why, how, and limitations of React. It provides a strong foundation in React, enabling you to write better, more optimal code by keeping the core principles in mind. It's important to remember that if you're a beginner, don't stress too much about writing the most optimal code right from day one. You'll learn and improve over time as you work on projects. Before diving into building components or features, make sure to thoroughly understand the requirements, just like how we tackle programming problems. Start with pseudo code and implement it using a brute force approach, and then optimize it later. Always remember the golden rule: "If it works, don't touch it π ." kidding But before starting optimization, make sure to have a backup of your working code. Happy coding! If you found this blog helpful, show your support for it. π
Upcoming Blogs
Understanding React Limitations
JavaScript Fundamentals for React
Comment topics that you would like to cover first I will give them priority.
βοΈ Please let me know in the comments if you prefer the current length of the blog or if you would like it to be shorter. Your feedback is valuable, and I want to ensure that the blog meets your expectations in terms of content length and readability. Thank you! π